/* * Copyright (C) 2013-2017 ARM Limited. All rights reserved. * * This program is free software and is provided to you under the terms of the GNU General Public License version 2 * as published by the Free Software Foundation, and any use by you of this program is subject to the terms of such GNU licence. * * A copy of the licence is included with the program, and can also be obtained from Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "mali_kernel_common.h" #include "mali_osk.h" #include "mali_ukk.h" #include "mali_memory.h" #include "mali_mem_validation.h" #include "mali_uk_types.h" void mali_mem_unbind_ext_buf(mali_mem_backend *mem_backend) { mali_mem_allocation *alloc; struct mali_session_data *session; MALI_DEBUG_ASSERT_POINTER(mem_backend); alloc = mem_backend->mali_allocation; MALI_DEBUG_ASSERT_POINTER(alloc); MALI_DEBUG_ASSERT(MALI_MEM_EXTERNAL == mem_backend->type); session = alloc->session; MALI_DEBUG_ASSERT_POINTER(session); mali_session_memory_lock(session); mali_mem_mali_map_free(session, alloc->psize, alloc->mali_vma_node.vm_node.start, alloc->flags); mali_session_memory_unlock(session); } _mali_osk_errcode_t mali_mem_bind_ext_buf(mali_mem_allocation *alloc, mali_mem_backend *mem_backend, u32 phys_addr, u32 flag) { struct mali_session_data *session; _mali_osk_errcode_t err; u32 virt, phys, size; MALI_DEBUG_ASSERT_POINTER(mem_backend); MALI_DEBUG_ASSERT_POINTER(alloc); size = alloc->psize; session = (struct mali_session_data *)(uintptr_t)alloc->session; MALI_CHECK_NON_NULL(session, _MALI_OSK_ERR_INVALID_ARGS); /* check arguments */ /* NULL might be a valid Mali address */ if (!size) MALI_ERROR(_MALI_OSK_ERR_INVALID_ARGS); /* size must be a multiple of the system page size */ if (size % _MALI_OSK_MALI_PAGE_SIZE) MALI_ERROR(_MALI_OSK_ERR_INVALID_ARGS); /* Validate the mali physical range */ if (_MALI_OSK_ERR_OK != mali_mem_validation_check(phys_addr, size)) { return _MALI_OSK_ERR_FAULT; } if (flag & _MALI_MAP_EXTERNAL_MAP_GUARD_PAGE) { alloc->flags |= MALI_MEM_FLAG_MALI_GUARD_PAGE; } mali_session_memory_lock(session); virt = alloc->mali_vma_node.vm_node.start; phys = phys_addr; err = mali_mem_mali_map_prepare(alloc); if (_MALI_OSK_ERR_OK != err) { mali_session_memory_unlock(session); return _MALI_OSK_ERR_NOMEM; } mali_mmu_pagedir_update(session->page_directory, virt, phys, size, MALI_MMU_FLAGS_DEFAULT); if (alloc->flags & MALI_MEM_FLAG_MALI_GUARD_PAGE) { mali_mmu_pagedir_update(session->page_directory, virt + size, phys, _MALI_OSK_MALI_PAGE_SIZE, MALI_MMU_FLAGS_DEFAULT); } MALI_DEBUG_PRINT(3, ("Requested to map physical memory 0x%x-0x%x into virtual memory 0x%x\n", phys_addr, (phys_addr + size - 1), virt)); mali_session_memory_unlock(session); MALI_SUCCESS; }