#include <linux/slab.h>
 #include <linux/udmabuf.h>
 #include <linux/hugetlb.h>
+#include <linux/vmalloc.h>
+#include <linux/iosys-map.h>
 
 static int list_limit = 1024;
 module_param(list_limit, int, 0644);
        return 0;
 }
 
+static int vmap_udmabuf(struct dma_buf *buf, struct iosys_map *map)
+{
+       struct udmabuf *ubuf = buf->priv;
+       void *vaddr;
+
+       dma_resv_assert_held(buf->resv);
+
+       vaddr = vm_map_ram(ubuf->pages, ubuf->pagecount, -1);
+       if (!vaddr)
+               return -EINVAL;
+
+       iosys_map_set_vaddr(map, vaddr);
+       return 0;
+}
+
+static void vunmap_udmabuf(struct dma_buf *buf, struct iosys_map *map)
+{
+       struct udmabuf *ubuf = buf->priv;
+
+       dma_resv_assert_held(buf->resv);
+
+       vm_unmap_ram(map->vaddr, ubuf->pagecount);
+}
+
 static struct sg_table *get_sg_table(struct device *dev, struct dma_buf *buf,
                                     enum dma_data_direction direction)
 {
        .unmap_dma_buf     = unmap_udmabuf,
        .release           = release_udmabuf,
        .mmap              = mmap_udmabuf,
+       .vmap              = vmap_udmabuf,
+       .vunmap            = vunmap_udmabuf,
        .begin_cpu_access  = begin_cpu_udmabuf,
        .end_cpu_access    = end_cpu_udmabuf,
 };