123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation; either version 2 of the License, or
- * (at your option) any later version.
- */
- #include <linux/shmem_fs.h>
- #include "vkms_drv.h"
- static struct vkms_gem_object *__vkms_gem_create(struct drm_device *dev,
- u64 size)
- {
- struct vkms_gem_object *obj;
- int ret;
- obj = kzalloc(sizeof(*obj), GFP_KERNEL);
- if (!obj)
- return ERR_PTR(-ENOMEM);
- size = roundup(size, PAGE_SIZE);
- ret = drm_gem_object_init(dev, &obj->gem, size);
- if (ret) {
- kfree(obj);
- return ERR_PTR(ret);
- }
- mutex_init(&obj->pages_lock);
- return obj;
- }
- void vkms_gem_free_object(struct drm_gem_object *obj)
- {
- struct vkms_gem_object *gem = container_of(obj, struct vkms_gem_object,
- gem);
- WARN_ON(gem->pages);
- WARN_ON(gem->vaddr);
- mutex_destroy(&gem->pages_lock);
- drm_gem_object_release(obj);
- kfree(gem);
- }
- vm_fault_t vkms_gem_fault(struct vm_fault *vmf)
- {
- struct vm_area_struct *vma = vmf->vma;
- struct vkms_gem_object *obj = vma->vm_private_data;
- unsigned long vaddr = vmf->address;
- pgoff_t page_offset;
- loff_t num_pages;
- vm_fault_t ret = VM_FAULT_SIGBUS;
- page_offset = (vaddr - vma->vm_start) >> PAGE_SHIFT;
- num_pages = DIV_ROUND_UP(obj->gem.size, PAGE_SIZE);
- if (page_offset > num_pages)
- return VM_FAULT_SIGBUS;
- mutex_lock(&obj->pages_lock);
- if (obj->pages) {
- get_page(obj->pages[page_offset]);
- vmf->page = obj->pages[page_offset];
- ret = 0;
- }
- mutex_unlock(&obj->pages_lock);
- if (ret) {
- struct page *page;
- struct address_space *mapping;
- mapping = file_inode(obj->gem.filp)->i_mapping;
- page = shmem_read_mapping_page(mapping, page_offset);
- if (!IS_ERR(page)) {
- vmf->page = page;
- ret = 0;
- } else {
- switch (PTR_ERR(page)) {
- case -ENOSPC:
- case -ENOMEM:
- ret = VM_FAULT_OOM;
- break;
- case -EBUSY:
- ret = VM_FAULT_RETRY;
- break;
- case -EFAULT:
- case -EINVAL:
- ret = VM_FAULT_SIGBUS;
- break;
- default:
- WARN_ON(PTR_ERR(page));
- ret = VM_FAULT_SIGBUS;
- break;
- }
- }
- }
- return ret;
- }
- struct drm_gem_object *vkms_gem_create(struct drm_device *dev,
- struct drm_file *file,
- u32 *handle,
- u64 size)
- {
- struct vkms_gem_object *obj;
- int ret;
- if (!file || !dev || !handle)
- return ERR_PTR(-EINVAL);
- obj = __vkms_gem_create(dev, size);
- if (IS_ERR(obj))
- return ERR_CAST(obj);
- ret = drm_gem_handle_create(file, &obj->gem, handle);
- drm_gem_object_put_unlocked(&obj->gem);
- if (ret) {
- drm_gem_object_release(&obj->gem);
- kfree(obj);
- return ERR_PTR(ret);
- }
- return &obj->gem;
- }
- int vkms_dumb_create(struct drm_file *file, struct drm_device *dev,
- struct drm_mode_create_dumb *args)
- {
- struct drm_gem_object *gem_obj;
- u64 pitch, size;
- if (!args || !dev || !file)
- return -EINVAL;
- pitch = args->width * DIV_ROUND_UP(args->bpp, 8);
- size = pitch * args->height;
- if (!size)
- return -EINVAL;
- gem_obj = vkms_gem_create(dev, file, &args->handle, size);
- if (IS_ERR(gem_obj))
- return PTR_ERR(gem_obj);
- args->size = gem_obj->size;
- args->pitch = pitch;
- DRM_DEBUG_DRIVER("Created object of size %lld\n", size);
- return 0;
- }
- int vkms_dumb_map(struct drm_file *file, struct drm_device *dev,
- u32 handle, u64 *offset)
- {
- struct drm_gem_object *obj;
- int ret;
- obj = drm_gem_object_lookup(file, handle);
- if (!obj)
- return -ENOENT;
- if (!obj->filp) {
- ret = -EINVAL;
- goto unref;
- }
- ret = drm_gem_create_mmap_offset(obj);
- if (ret)
- goto unref;
- *offset = drm_vma_node_offset_addr(&obj->vma_node);
- unref:
- drm_gem_object_put_unlocked(obj);
- return ret;
- }
- static struct page **_get_pages(struct vkms_gem_object *vkms_obj)
- {
- struct drm_gem_object *gem_obj = &vkms_obj->gem;
- if (!vkms_obj->pages) {
- struct page **pages = drm_gem_get_pages(gem_obj);
- if (IS_ERR(pages))
- return pages;
- if (cmpxchg(&vkms_obj->pages, NULL, pages))
- drm_gem_put_pages(gem_obj, pages, false, true);
- }
- return vkms_obj->pages;
- }
- void vkms_gem_vunmap(struct drm_gem_object *obj)
- {
- struct vkms_gem_object *vkms_obj = drm_gem_to_vkms_gem(obj);
- mutex_lock(&vkms_obj->pages_lock);
- if (vkms_obj->vmap_count < 1) {
- WARN_ON(vkms_obj->vaddr);
- WARN_ON(vkms_obj->pages);
- mutex_unlock(&vkms_obj->pages_lock);
- return;
- }
- vkms_obj->vmap_count--;
- if (vkms_obj->vmap_count == 0) {
- vunmap(vkms_obj->vaddr);
- vkms_obj->vaddr = NULL;
- drm_gem_put_pages(obj, vkms_obj->pages, false, true);
- vkms_obj->pages = NULL;
- }
- mutex_unlock(&vkms_obj->pages_lock);
- }
- int vkms_gem_vmap(struct drm_gem_object *obj)
- {
- struct vkms_gem_object *vkms_obj = drm_gem_to_vkms_gem(obj);
- int ret = 0;
- mutex_lock(&vkms_obj->pages_lock);
- if (!vkms_obj->vaddr) {
- unsigned int n_pages = obj->size >> PAGE_SHIFT;
- struct page **pages = _get_pages(vkms_obj);
- if (IS_ERR(pages)) {
- ret = PTR_ERR(pages);
- goto out;
- }
- vkms_obj->vaddr = vmap(pages, n_pages, VM_MAP, PAGE_KERNEL);
- if (!vkms_obj->vaddr)
- goto err_vmap;
- }
- vkms_obj->vmap_count++;
- goto out;
- err_vmap:
- ret = -ENOMEM;
- drm_gem_put_pages(obj, vkms_obj->pages, false, true);
- vkms_obj->pages = NULL;
- out:
- mutex_unlock(&vkms_obj->pages_lock);
- return ret;
- }
|