Fichiers
rocm-systems/src/memory.c
T

195 lignes
5.2 KiB
C
Brut Vue normale Historique

/*
* Copyright © 2014 Advanced Micro Devices, Inc.
*
* Permission is hereby granted, free of charge, to any person
* obtaining a copy of this software and associated documentation
* files (the "Software"), to deal in the Software without
* restriction, including without limitation the rights to use, copy,
* modify, merge, publish, distribute, sublicense, and/or sell copies
* of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice (including
* the next paragraph) shall be included in all copies or substantial
* portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*/
#include "libhsakmt.h"
#include "linux/kfd_ioctl.h"
#include <stdlib.h>
#include <string.h>
#include <assert.h>
2014-07-29 13:56:31 +03:00
#include "fmm.h"
HSAKMT_STATUS
HSAKMTAPI
hsaKmtSetMemoryPolicy(
HSAuint32 Node,
HSAuint32 DefaultPolicy,
HSAuint32 AlternatePolicy,
void* MemoryAddressAlternate,
HSAuint64 MemorySizeInBytes
)
{
2014-07-29 13:56:31 +03:00
HSAKMT_STATUS result;
uint32_t gpu_id;
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
result = validate_nodeid(Node, &gpu_id);
if (result != HSAKMT_STATUS_SUCCESS)
return result;
// We accept any legal policy and alternate address location. You get CC everywhere anyway.
if ((DefaultPolicy != HSA_CACHING_CACHED && DefaultPolicy != HSA_CACHING_NONCACHED)
|| (AlternatePolicy != HSA_CACHING_CACHED && AlternatePolicy != HSA_CACHING_NONCACHED))
{
return HSAKMT_STATUS_INVALID_PARAMETER;
}
CHECK_PAGE_MULTIPLE(MemoryAddressAlternate);
CHECK_PAGE_MULTIPLE(MemorySizeInBytes);
struct kfd_ioctl_set_memory_policy_args args;
memset(&args, 0, sizeof(args));
args.gpu_id = gpu_id;
args.default_policy = (DefaultPolicy == HSA_CACHING_CACHED) ? KFD_IOC_CACHE_POLICY_COHERENT : KFD_IOC_CACHE_POLICY_NONCOHERENT;
args.alternate_policy = (AlternatePolicy == HSA_CACHING_CACHED) ? KFD_IOC_CACHE_POLICY_COHERENT : KFD_IOC_CACHE_POLICY_NONCOHERENT;
args.alternate_aperture_base = (uintptr_t)MemoryAddressAlternate;
args.alternate_aperture_size = MemorySizeInBytes;
2014-12-29 13:34:32 +02:00
int err = kfd_ioctl(AMDKFD_IOC_SET_MEMORY_POLICY, &args);
2014-07-29 13:56:31 +03:00
return (err == -1) ? HSAKMT_STATUS_ERROR : HSAKMT_STATUS_SUCCESS;
}
static HSAuint32 PageSizeFromFlags(unsigned int pageSizeFlags)
{
switch (pageSizeFlags)
{
case HSA_PAGE_SIZE_4KB: return 4*1024;
case HSA_PAGE_SIZE_64KB: return 64*1024;
case HSA_PAGE_SIZE_2MB: return 2*1024*1024;
case HSA_PAGE_SIZE_1GB: return 1024*1024*1024;
default: assert(false); return 4*1024;
}
}
HSAKMT_STATUS
HSAKMTAPI
hsaKmtAllocMemory(
HSAuint32 PreferredNode, //IN
HSAuint64 SizeInBytes, //IN (multiple of page size)
HsaMemFlags MemFlags, //IN
void** MemoryAddress //OUT (page-aligned)
)
{
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
HSAKMT_STATUS result;
uint32_t gpu_id;
result = validate_nodeid(PreferredNode, &gpu_id);
if (result != HSAKMT_STATUS_SUCCESS)
return result;
// The required size should be page aligned (GDS?)
HSAuint64 page_size = PageSizeFromFlags(MemFlags.ui32.PageSize);
if ((SizeInBytes & (page_size-1)) && !MemFlags.ui32.GDSMemory){
return HSAKMT_STATUS_INVALID_PARAMETER;
}
if (MemFlags.ui32.HostAccess && !MemFlags.ui32.NonPaged){
int err = posix_memalign(MemoryAddress, page_size, SizeInBytes);
if (err == 0)
return HSAKMT_STATUS_SUCCESS;
else
return HSAKMT_STATUS_NO_MEMORY;
}
else
return HSAKMT_STATUS_INVALID_PARAMETER;
}
HSAKMT_STATUS
HSAKMTAPI
hsaKmtFreeMemory(
void* MemoryAddress, //IN (page-aligned)
HSAuint64 SizeInBytes //IN
)
{
2014-07-29 13:56:31 +03:00
HSAKMT_STATUS hsa_status = HSAKMT_STATUS_SUCCESS;
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
if (fmm_is_inside_some_aperture(MemoryAddress)){
if (fmm_release( MemoryAddress, SizeInBytes))
hsa_status = HSAKMT_STATUS_INVALID_PARAMETER;
}
else
free(MemoryAddress);
return hsa_status;
}
HSAKMT_STATUS
HSAKMTAPI
hsaKmtRegisterMemory(
void* MemoryAddress, //IN (page-aligned)
HSAuint64 MemorySizeInBytes //IN (page-aligned)
)
{
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
return HSAKMT_STATUS_SUCCESS;
}
HSAKMT_STATUS
HSAKMTAPI
hsaKmtDeregisterMemory(
void* MemoryAddress //IN
)
{
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
return HSAKMT_STATUS_SUCCESS;
}
HSAKMT_STATUS
HSAKMTAPI
hsaKmtMapMemoryToGPU(
void* MemoryAddress, //IN (page-aligned)
HSAuint64 MemorySizeInBytes, //IN (page-aligned)
HSAuint64* AlternateVAGPU //OUT (page-aligned)
)
{
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
// We don't support GPUVM in the stub, there should never be a request for a GPUVA.
if (AlternateVAGPU)
{
*AlternateVAGPU = 0;
}
return HSAKMT_STATUS_SUCCESS;
}
HSAKMT_STATUS
HSAKMTAPI
hsaKmtUnmapMemoryToGPU(
void* MemoryAddress //IN (page-aligned)
)
{
CHECK_KFD_OPEN();
2014-07-29 13:56:31 +03:00
return HSAKMT_STATUS_SUCCESS;
}