/* * Copyright (c) 2017, NVIDIA CORPORATION. All rights reserved. * * 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 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 "nvidia-drm-conftest.h" #if defined(NV_DRM_ATOMIC_MODESET_AVAILABLE) #include "nvidia-drm-gem-nvkms-memory.h" #include "nvidia-drm-helper.h" #include "nvidia-drm-ioctl.h" #if defined(NV_DRM_DRM_DRV_H_PRESENT) #include #endif #if defined(NV_DRM_DRM_PRIME_H_PRESENT) #include #endif #include #if defined(NV_BSD) #include #endif #include "nv-mm.h" static void __nv_drm_gem_nvkms_memory_free(struct nv_drm_gem_object *nv_gem) { struct nv_drm_device *nv_dev = nv_gem->nv_dev; struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = to_nv_nvkms_memory(nv_gem); if (nv_nvkms_memory->physically_mapped) { if (nv_nvkms_memory->pWriteCombinedIORemapAddress != NULL) { iounmap(nv_nvkms_memory->pWriteCombinedIORemapAddress); } nvKms->unmapMemory(nv_dev->pDevice, nv_nvkms_memory->base.pMemory, NVKMS_KAPI_MAPPING_TYPE_USER, nv_nvkms_memory->pPhysicalAddress); } if (nv_nvkms_memory->pages_count != 0) { nvKms->freeMemoryPages((NvU64 *)nv_nvkms_memory->pages); } /* Free NvKmsKapiMemory handle associated with this gem object */ nvKms->freeMemory(nv_dev->pDevice, nv_nvkms_memory->base.pMemory); nv_drm_free(nv_nvkms_memory); } static int __nv_drm_gem_nvkms_mmap(struct nv_drm_gem_object *nv_gem, struct vm_area_struct *vma) { return drm_gem_mmap_obj(&nv_gem->base, drm_vma_node_size(&nv_gem->base.vma_node) << PAGE_SHIFT, vma); } static vm_fault_t __nv_drm_gem_nvkms_handle_vma_fault( struct nv_drm_gem_object *nv_gem, struct vm_area_struct *vma, struct vm_fault *vmf) { #if defined(NV_DRM_ATOMIC_MODESET_AVAILABLE) struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = to_nv_nvkms_memory(nv_gem); unsigned long address = nv_page_fault_va(vmf); struct drm_gem_object *gem = vma->vm_private_data; unsigned long page_offset, pfn; vm_fault_t ret; page_offset = vmf->pgoff - drm_vma_node_start(&gem->vma_node); if (nv_nvkms_memory->pages_count == 0) { pfn = (unsigned long)(uintptr_t)nv_nvkms_memory->pPhysicalAddress; pfn >>= PAGE_SHIFT; #if defined(NV_LINUX) /* * FreeBSD doesn't set pgoff. We instead have pfn be the base physical * address, and we will calculate the index pidx from the virtual address. * * This only works because linux_cdev_pager_populate passes the pidx as * vmf->virtual_address. Then we turn the virtual address * into a physical page number. */ pfn += page_offset; #endif } else { BUG_ON(page_offset >= nv_nvkms_memory->pages_count); pfn = page_to_pfn(nv_nvkms_memory->pages[page_offset]); } #if defined(NV_VMF_INSERT_PFN_PRESENT) ret = vmf_insert_pfn(vma, address, pfn); #else ret = vm_insert_pfn(vma, address, pfn); switch (ret) { case 0: case -EBUSY: /* * EBUSY indicates that another thread already handled * the faulted range. */ ret = VM_FAULT_NOPAGE; break; case -ENOMEM: ret = VM_FAULT_OOM; break; default: WARN_ONCE(1, "Unhandled error in %s: %d\n", __FUNCTION__, ret); ret = VM_FAULT_SIGBUS; break; } #endif /* defined(NV_VMF_INSERT_PFN_PRESENT) */ return ret; #endif /* defined(NV_DRM_ATOMIC_MODESET_AVAILABLE) */ return VM_FAULT_SIGBUS; } static struct drm_gem_object *__nv_drm_gem_nvkms_prime_dup( struct drm_device *dev, const struct nv_drm_gem_object *nv_gem_src); static int __nv_drm_gem_nvkms_map( struct nv_drm_gem_nvkms_memory *nv_nvkms_memory) { struct nv_drm_device *nv_dev = nv_nvkms_memory->base.nv_dev; struct NvKmsKapiMemory *pMemory = nv_nvkms_memory->base.pMemory; if (!nv_dev->hasVideoMemory) { return 0; } if (!nvKms->mapMemory(nv_dev->pDevice, pMemory, NVKMS_KAPI_MAPPING_TYPE_USER, &nv_nvkms_memory->pPhysicalAddress)) { NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to map NvKmsKapiMemory 0x%p", pMemory); return -ENOMEM; } nv_nvkms_memory->pWriteCombinedIORemapAddress = ioremap_wc( (uintptr_t)nv_nvkms_memory->pPhysicalAddress, nv_nvkms_memory->base.base.size); if (!nv_nvkms_memory->pWriteCombinedIORemapAddress) { NV_DRM_DEV_LOG_INFO( nv_dev, "Failed to ioremap_wc NvKmsKapiMemory 0x%p", pMemory); } nv_nvkms_memory->physically_mapped = true; return 0; } static void *__nv_drm_gem_nvkms_prime_vmap( struct nv_drm_gem_object *nv_gem) { struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = to_nv_nvkms_memory(nv_gem); if (!nv_nvkms_memory->physically_mapped) { int ret = __nv_drm_gem_nvkms_map(nv_nvkms_memory); if (ret) { return ERR_PTR(ret); } } return nv_nvkms_memory->pWriteCombinedIORemapAddress; } static int __nv_drm_gem_map_nvkms_memory_offset( struct nv_drm_device *nv_dev, struct nv_drm_gem_object *nv_gem, uint64_t *offset) { struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = to_nv_nvkms_memory(nv_gem); if (!nv_nvkms_memory->physically_mapped) { int ret = __nv_drm_gem_nvkms_map(nv_nvkms_memory); if (ret) { return ret; } } return nv_drm_gem_create_mmap_offset(&nv_nvkms_memory->base, offset); } static struct sg_table *__nv_drm_gem_nvkms_memory_prime_get_sg_table( struct nv_drm_gem_object *nv_gem) { struct nv_drm_device *nv_dev = nv_gem->nv_dev; struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = to_nv_nvkms_memory(nv_gem); struct sg_table *sg_table; if (nv_nvkms_memory->pages_count == 0) { NV_DRM_DEV_LOG_ERR( nv_dev, "Cannot create sg_table for NvKmsKapiMemory 0x%p", nv_gem->pMemory); return ERR_PTR(-ENOMEM); } sg_table = nv_drm_prime_pages_to_sg(nv_dev->dev, nv_nvkms_memory->pages, nv_nvkms_memory->pages_count); return sg_table; } const struct nv_drm_gem_object_funcs nv_gem_nvkms_memory_ops = { .free = __nv_drm_gem_nvkms_memory_free, .prime_dup = __nv_drm_gem_nvkms_prime_dup, .prime_vmap = __nv_drm_gem_nvkms_prime_vmap, .mmap = __nv_drm_gem_nvkms_mmap, .handle_vma_fault = __nv_drm_gem_nvkms_handle_vma_fault, .create_mmap_offset = __nv_drm_gem_map_nvkms_memory_offset, .prime_get_sg_table = __nv_drm_gem_nvkms_memory_prime_get_sg_table, }; static int __nv_drm_nvkms_gem_obj_init( struct nv_drm_device *nv_dev, struct nv_drm_gem_nvkms_memory *nv_nvkms_memory, struct NvKmsKapiMemory *pMemory, uint64_t size) { NvU64 *pages = NULL; NvU32 numPages = 0; if ((size % PAGE_SIZE) != 0) { NV_DRM_DEV_LOG_ERR( nv_dev, "NvKmsKapiMemory 0x%p size should be in a multiple of page size to " "create a gem object", pMemory); return -EINVAL; } nv_nvkms_memory->pPhysicalAddress = NULL; nv_nvkms_memory->pWriteCombinedIORemapAddress = NULL; nv_nvkms_memory->physically_mapped = false; if (!nvKms->getMemoryPages(nv_dev->pDevice, pMemory, &pages, &numPages) && !nv_dev->hasVideoMemory) { /* GetMemoryPages may fail for vidmem allocations, * but it should not fail for sysmem allocations. */ NV_DRM_DEV_LOG_ERR(nv_dev, "Failed to get memory pages for NvKmsKapiMemory 0x%p", pMemory); return -ENOMEM; } nv_nvkms_memory->pages_count = numPages; nv_nvkms_memory->pages = (struct page **)pages; nv_drm_gem_object_init(nv_dev, &nv_nvkms_memory->base, &nv_gem_nvkms_memory_ops, size, pMemory); return 0; } int nv_drm_dumb_create( struct drm_file *file_priv, struct drm_device *dev, struct drm_mode_create_dumb *args) { struct nv_drm_device *nv_dev = to_nv_device(dev); struct nv_drm_gem_nvkms_memory *nv_nvkms_memory; uint8_t compressible = 0; struct NvKmsKapiMemory *pMemory; int ret = 0; args->pitch = roundup(args->width * ((args->bpp + 7) >> 3), nv_dev->pitchAlignment); args->size = args->height * args->pitch; /* Core DRM requires gem object size to be aligned with PAGE_SIZE */ args->size = roundup(args->size, PAGE_SIZE); if ((nv_nvkms_memory = nv_drm_calloc(1, sizeof(*nv_nvkms_memory))) == NULL) { ret = -ENOMEM; goto fail; } if (nv_dev->hasVideoMemory) { pMemory = nvKms->allocateVideoMemory(nv_dev->pDevice, NvKmsSurfaceMemoryLayoutPitch, NVKMS_KAPI_ALLOCATION_TYPE_SCANOUT, args->size, &compressible); } else { pMemory = nvKms->allocateSystemMemory(nv_dev->pDevice, NvKmsSurfaceMemoryLayoutPitch, NVKMS_KAPI_ALLOCATION_TYPE_SCANOUT, args->size, &compressible); } if (pMemory == NULL) { ret = -ENOMEM; NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to allocate NvKmsKapiMemory for dumb object of size %" NvU64_fmtu, args->size); goto nvkms_alloc_memory_failed; } ret = __nv_drm_nvkms_gem_obj_init(nv_dev, nv_nvkms_memory, pMemory, args->size); if (ret) { goto nvkms_gem_obj_init_failed; } /* Always map dumb buffer memory up front. Clients are only expected * to use dumb buffers for software rendering, so they're not much use * without a CPU mapping. */ ret = __nv_drm_gem_nvkms_map(nv_nvkms_memory); if (ret) { nv_drm_gem_object_unreference_unlocked(&nv_nvkms_memory->base); goto fail; } return nv_drm_gem_handle_create_drop_reference(file_priv, &nv_nvkms_memory->base, &args->handle); nvkms_gem_obj_init_failed: nvKms->freeMemory(nv_dev->pDevice, pMemory); nvkms_alloc_memory_failed: nv_drm_free(nv_nvkms_memory); fail: return ret; } int nv_drm_gem_import_nvkms_memory_ioctl(struct drm_device *dev, void *data, struct drm_file *filep) { struct nv_drm_device *nv_dev = to_nv_device(dev); struct drm_nvidia_gem_import_nvkms_memory_params *p = data; struct nv_drm_gem_nvkms_memory *nv_nvkms_memory; struct NvKmsKapiMemory *pMemory; int ret; if (!drm_core_check_feature(dev, DRIVER_MODESET)) { ret = -EOPNOTSUPP; goto failed; } if ((nv_nvkms_memory = nv_drm_calloc(1, sizeof(*nv_nvkms_memory))) == NULL) { ret = -ENOMEM; goto failed; } pMemory = nvKms->importMemory(nv_dev->pDevice, p->mem_size, p->nvkms_params_ptr, p->nvkms_params_size); if (pMemory == NULL) { ret = -EINVAL; NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to import NVKMS memory to GEM object"); goto nvkms_import_memory_failed; } ret = __nv_drm_nvkms_gem_obj_init(nv_dev, nv_nvkms_memory, pMemory, p->mem_size); if (ret) { goto nvkms_gem_obj_init_failed; } return nv_drm_gem_handle_create_drop_reference(filep, &nv_nvkms_memory->base, &p->handle); nvkms_gem_obj_init_failed: nvKms->freeMemory(nv_dev->pDevice, pMemory); nvkms_import_memory_failed: nv_drm_free(nv_nvkms_memory); failed: return ret; } int nv_drm_gem_export_nvkms_memory_ioctl(struct drm_device *dev, void *data, struct drm_file *filep) { struct nv_drm_device *nv_dev = to_nv_device(dev); struct drm_nvidia_gem_export_nvkms_memory_params *p = data; struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = NULL; int ret = 0; if (!drm_core_check_feature(dev, DRIVER_MODESET)) { ret = -EOPNOTSUPP; goto done; } if (p->__pad != 0) { ret = -EINVAL; NV_DRM_DEV_LOG_ERR(nv_dev, "Padding fields must be zeroed"); goto done; } if ((nv_nvkms_memory = nv_drm_gem_object_nvkms_memory_lookup( dev, filep, p->handle)) == NULL) { ret = -EINVAL; NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to lookup NVKMS gem object for export: 0x%08x", p->handle); goto done; } if (!nvKms->exportMemory(nv_dev->pDevice, nv_nvkms_memory->base.pMemory, p->nvkms_params_ptr, p->nvkms_params_size)) { ret = -EINVAL; NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to export memory from NVKMS GEM object: 0x%08x", p->handle); goto done; } done: if (nv_nvkms_memory != NULL) { nv_drm_gem_object_unreference_unlocked(&nv_nvkms_memory->base); } return ret; } int nv_drm_gem_alloc_nvkms_memory_ioctl(struct drm_device *dev, void *data, struct drm_file *filep) { struct nv_drm_device *nv_dev = to_nv_device(dev); struct drm_nvidia_gem_alloc_nvkms_memory_params *p = data; struct nv_drm_gem_nvkms_memory *nv_nvkms_memory = NULL; struct NvKmsKapiMemory *pMemory; enum NvKmsSurfaceMemoryLayout layout; enum NvKmsKapiAllocationType type; int ret = 0; if (!drm_core_check_feature(dev, DRIVER_MODESET)) { ret = -EOPNOTSUPP; goto failed; } if ((p->__pad0 != 0) || (p->__pad1 != 0)) { ret = -EINVAL; NV_DRM_DEV_LOG_ERR(nv_dev, "non-zero value in padding field"); goto failed; } if ((nv_nvkms_memory = nv_drm_calloc(1, sizeof(*nv_nvkms_memory))) == NULL) { ret = -ENOMEM; goto failed; } layout = p->block_linear ? NvKmsSurfaceMemoryLayoutBlockLinear : NvKmsSurfaceMemoryLayoutPitch; type = (p->flags & NV_GEM_ALLOC_NO_SCANOUT) ? NVKMS_KAPI_ALLOCATION_TYPE_OFFSCREEN : NVKMS_KAPI_ALLOCATION_TYPE_SCANOUT; if (nv_dev->hasVideoMemory) { pMemory = nvKms->allocateVideoMemory(nv_dev->pDevice, layout, type, p->memory_size, &p->compressible); } else { pMemory = nvKms->allocateSystemMemory(nv_dev->pDevice, layout, type, p->memory_size, &p->compressible); } if (pMemory == NULL) { ret = -EINVAL; NV_DRM_DEV_LOG_ERR(nv_dev, "Failed to allocate NVKMS memory for GEM object"); goto nvkms_alloc_memory_failed; } ret = __nv_drm_nvkms_gem_obj_init(nv_dev, nv_nvkms_memory, pMemory, p->memory_size); if (ret) { goto nvkms_gem_obj_init_failed; } return nv_drm_gem_handle_create_drop_reference(filep, &nv_nvkms_memory->base, &p->handle); nvkms_gem_obj_init_failed: nvKms->freeMemory(nv_dev->pDevice, pMemory); nvkms_alloc_memory_failed: nv_drm_free(nv_nvkms_memory); failed: return ret; } static struct drm_gem_object *__nv_drm_gem_nvkms_prime_dup( struct drm_device *dev, const struct nv_drm_gem_object *nv_gem_src) { struct nv_drm_device *nv_dev = to_nv_device(dev); const struct nv_drm_device *nv_dev_src; struct nv_drm_gem_nvkms_memory *nv_nvkms_memory; struct NvKmsKapiMemory *pMemory; BUG_ON(nv_gem_src == NULL || nv_gem_src->ops != &nv_gem_nvkms_memory_ops); nv_dev_src = to_nv_device(nv_gem_src->base.dev); if ((nv_nvkms_memory = nv_drm_calloc(1, sizeof(*nv_nvkms_memory))) == NULL) { return NULL; } pMemory = nvKms->dupMemory(nv_dev->pDevice, nv_dev_src->pDevice, nv_gem_src->pMemory); if (pMemory == NULL) { NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to import NVKMS memory to GEM object"); goto nvkms_dup_memory_failed; } if (__nv_drm_nvkms_gem_obj_init(nv_dev, nv_nvkms_memory, pMemory, nv_gem_src->base.size)) { goto nvkms_gem_obj_init_failed; } return &nv_nvkms_memory->base.base; nvkms_gem_obj_init_failed: nvKms->freeMemory(nv_dev->pDevice, pMemory); nvkms_dup_memory_failed: nv_drm_free(nv_nvkms_memory); return NULL; } int nv_drm_dumb_map_offset(struct drm_file *file, struct drm_device *dev, uint32_t handle, uint64_t *offset) { struct nv_drm_device *nv_dev = to_nv_device(dev); struct nv_drm_gem_nvkms_memory *nv_nvkms_memory; int ret = -EINVAL; if ((nv_nvkms_memory = nv_drm_gem_object_nvkms_memory_lookup( dev, file, handle)) == NULL) { NV_DRM_DEV_LOG_ERR( nv_dev, "Failed to lookup gem object for mapping: 0x%08x", handle); return ret; } ret = __nv_drm_gem_map_nvkms_memory_offset(nv_dev, &nv_nvkms_memory->base, offset); nv_drm_gem_object_unreference_unlocked(&nv_nvkms_memory->base); return ret; } #if defined(NV_DRM_DRIVER_HAS_DUMB_DESTROY) int nv_drm_dumb_destroy(struct drm_file *file, struct drm_device *dev, uint32_t handle) { return drm_gem_handle_delete(file, handle); } #endif /* NV_DRM_DRIVER_HAS_DUMB_DESTROY */ #endif