/*
|
* Copyright (C) 2012-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_ukk.h"
|
#include "mali_osk.h"
|
#include "mali_kernel_common.h"
|
#include "mali_session.h"
|
#include "mali_kernel_linux.h"
|
#include "mali_memory.h"
|
#include "ump_kernel_interface.h"
|
|
static int mali_mem_ump_map(mali_mem_backend *mem_backend)
|
{
|
ump_dd_handle ump_mem;
|
mali_mem_allocation *alloc;
|
struct mali_session_data *session;
|
u32 nr_blocks;
|
u32 i;
|
ump_dd_physical_block *ump_blocks;
|
struct mali_page_directory *pagedir;
|
u32 offset = 0;
|
_mali_osk_errcode_t err;
|
|
MALI_DEBUG_ASSERT_POINTER(mem_backend);
|
MALI_DEBUG_ASSERT(MALI_MEM_UMP == mem_backend->type);
|
|
alloc = mem_backend->mali_allocation;
|
MALI_DEBUG_ASSERT_POINTER(alloc);
|
|
session = alloc->session;
|
MALI_DEBUG_ASSERT_POINTER(session);
|
|
ump_mem = mem_backend->ump_mem.handle;
|
MALI_DEBUG_ASSERT(UMP_DD_HANDLE_INVALID != ump_mem);
|
|
nr_blocks = ump_dd_phys_block_count_get(ump_mem);
|
if (nr_blocks == 0) {
|
MALI_DEBUG_PRINT(1, ("No block count\n"));
|
return -EINVAL;
|
}
|
|
ump_blocks = _mali_osk_malloc(sizeof(*ump_blocks) * nr_blocks);
|
if (NULL == ump_blocks) {
|
return -ENOMEM;
|
}
|
|
if (UMP_DD_INVALID == ump_dd_phys_blocks_get(ump_mem, ump_blocks, nr_blocks)) {
|
_mali_osk_free(ump_blocks);
|
return -EFAULT;
|
}
|
|
pagedir = session->page_directory;
|
|
mali_session_memory_lock(session);
|
|
err = mali_mem_mali_map_prepare(alloc);
|
if (_MALI_OSK_ERR_OK != err) {
|
MALI_DEBUG_PRINT(1, ("Mapping of UMP memory failed\n"));
|
|
_mali_osk_free(ump_blocks);
|
mali_session_memory_unlock(session);
|
return -ENOMEM;
|
}
|
|
for (i = 0; i < nr_blocks; ++i) {
|
u32 virt = alloc->mali_vma_node.vm_node.start + offset;
|
|
MALI_DEBUG_PRINT(7, ("Mapping in 0x%08x size %d\n", ump_blocks[i].addr , ump_blocks[i].size));
|
|
mali_mmu_pagedir_update(pagedir, virt, ump_blocks[i].addr,
|
ump_blocks[i].size, MALI_MMU_FLAGS_DEFAULT);
|
|
offset += ump_blocks[i].size;
|
}
|
|
if (alloc->flags & _MALI_MAP_EXTERNAL_MAP_GUARD_PAGE) {
|
u32 virt = alloc->mali_vma_node.vm_node.start + offset;
|
|
/* Map in an extra virtual guard page at the end of the VMA */
|
MALI_DEBUG_PRINT(6, ("Mapping in extra guard page\n"));
|
|
mali_mmu_pagedir_update(pagedir, virt, ump_blocks[0].addr, _MALI_OSK_MALI_PAGE_SIZE, MALI_MMU_FLAGS_DEFAULT);
|
|
offset += _MALI_OSK_MALI_PAGE_SIZE;
|
}
|
mali_session_memory_unlock(session);
|
_mali_osk_free(ump_blocks);
|
return 0;
|
}
|
|
static void mali_mem_ump_unmap(mali_mem_allocation *alloc)
|
{
|
struct mali_session_data *session;
|
MALI_DEBUG_ASSERT_POINTER(alloc);
|
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);
|
}
|
|
int mali_mem_bind_ump_buf(mali_mem_allocation *alloc, mali_mem_backend *mem_backend, u32 secure_id, u32 flags)
|
{
|
ump_dd_handle ump_mem;
|
int ret;
|
MALI_DEBUG_ASSERT_POINTER(alloc);
|
MALI_DEBUG_ASSERT_POINTER(mem_backend);
|
MALI_DEBUG_ASSERT(MALI_MEM_UMP == mem_backend->type);
|
|
MALI_DEBUG_PRINT(3,
|
("Requested to map ump memory with secure id %d into virtual memory 0x%08X, size 0x%08X\n",
|
secure_id, alloc->mali_vma_node.vm_node.start, alloc->mali_vma_node.vm_node.size));
|
|
ump_mem = ump_dd_handle_create_from_secure_id(secure_id);
|
if (UMP_DD_HANDLE_INVALID == ump_mem) MALI_ERROR(_MALI_OSK_ERR_FAULT);
|
alloc->flags |= MALI_MEM_FLAG_DONT_CPU_MAP;
|
if (flags & _MALI_MAP_EXTERNAL_MAP_GUARD_PAGE) {
|
alloc->flags |= MALI_MEM_FLAG_MALI_GUARD_PAGE;
|
}
|
|
mem_backend->ump_mem.handle = ump_mem;
|
|
ret = mali_mem_ump_map(mem_backend);
|
if (0 != ret) {
|
ump_dd_reference_release(ump_mem);
|
return _MALI_OSK_ERR_FAULT;
|
}
|
MALI_DEBUG_PRINT(3, ("Returning from UMP bind\n"));
|
return _MALI_OSK_ERR_OK;
|
}
|
|
void mali_mem_unbind_ump_buf(mali_mem_backend *mem_backend)
|
{
|
ump_dd_handle ump_mem;
|
mali_mem_allocation *alloc;
|
MALI_DEBUG_ASSERT_POINTER(mem_backend);
|
MALI_DEBUG_ASSERT(MALI_MEM_UMP == mem_backend->type);
|
ump_mem = mem_backend->ump_mem.handle;
|
MALI_DEBUG_ASSERT(UMP_DD_HANDLE_INVALID != ump_mem);
|
|
alloc = mem_backend->mali_allocation;
|
MALI_DEBUG_ASSERT_POINTER(alloc);
|
mali_mem_ump_unmap(alloc);
|
ump_dd_reference_release(ump_mem);
|
}
|