libnx/nx/source/nvidia/map.c
fincs 5fe01c065a Major refactor and redesign of nvidia wrapper objects, see details:
- NvBuffer replaced with NvMap, which only manages the creation of
  raw nvmap objects. Users must map these objects manually to
  address spaces.
- nvAddressSpaceBindToChannel removed.
- nvChannelSetNvmapFd is now automatic and has been removed.
- Nv3DContext, NvCmdList, NvErrorNotifier, NvGpfifo, NvGpu and
  NvZcullContext have all been removed.
- Introduced NvGpuChannel, which provides all functionality that was
  part of said removed objects.
- Other miscellaneous changes and fixes.
2018-12-19 19:33:31 +01:00

103 lines
2.2 KiB
C

#include <malloc.h>
#include "types.h"
#include "result.h"
#include "arm/atomics.h"
#include "arm/cache.h"
#include "kernel/svc.h"
#include "services/nv.h"
#include "nvidia/ioctl.h"
#include "nvidia/map.h"
static u32 g_nvmap_fd = -1;
static u64 g_refCnt;
Result nvMapInit(void)
{
Result rc;
if (atomicIncrement64(&g_refCnt) > 0)
return 0;
rc = nvOpen(&g_nvmap_fd, "/dev/nvmap");
if (R_FAILED(rc))
atomicDecrement64(&g_refCnt);
return rc;
}
void nvMapExit(void)
{
if (atomicDecrement64(&g_refCnt) == 0)
{
if (g_nvmap_fd != -1)
nvClose(g_nvmap_fd);
g_nvmap_fd = -1;
}
}
u32 nvMapGetFd(void)
{
return g_nvmap_fd;
}
Result nvMapCreate(NvMap* m, void* cpu_addr, u32 size, u32 align, NvKind kind, bool is_cpu_cacheable)
{
Result rc;
if (align < 0x1000)
align = 0x1000;
if (align & (align-1))
return MAKERESULT(Module_Libnx, LibnxError_BadInput);
if (!size || (size & 0xFFF))
return MAKERESULT(Module_Libnx, LibnxError_BadInput);
if (!cpu_addr || ((uintptr_t)cpu_addr & 0xFFF))
return MAKERESULT(Module_Libnx, LibnxError_BadInput);
m->has_init = true;
m->is_cpu_cacheable = is_cpu_cacheable;
m->size = size;
m->handle = -1;
m->cpu_addr = cpu_addr;
m->kind = kind;
rc = nvioctlNvmap_Create(g_nvmap_fd, size, &m->handle);
if (R_SUCCEEDED(rc))
rc = nvioctlNvmap_Alloc(g_nvmap_fd, m->handle,
0, is_cpu_cacheable ? 1 : 0, align, kind, m->cpu_addr);
if (R_SUCCEEDED(rc) && !is_cpu_cacheable) {
armDCacheFlush(m->cpu_addr, m->size);
svcSetMemoryAttribute(m->cpu_addr, m->size, 8, 8);
}
if (R_SUCCEEDED(rc))
rc = nvioctlNvmap_GetId(g_nvmap_fd, m->handle, &m->id);
if (R_FAILED(rc))
nvMapFree(m);
return rc;
}
void nvMapFree(NvMap* m)
{
if (!m->has_init)
return;
if (m->handle != -1) {
nvioctlNvmap_Free(g_nvmap_fd, m->handle);
m->handle = -1;
}
if (m->cpu_addr) {
if (!m->is_cpu_cacheable)
svcSetMemoryAttribute(m->cpu_addr, m->size, 8, 0);
m->cpu_addr = NULL;
}
m->has_init = false;
}