**************************************************************************/
 
 #include "vmwgfx_drv.h"
+
+#include "vmwgfx_bo.h"
 #include <linux/highmem.h>
 
 /*
        return 0;
 }
 
+static void *map_external(struct vmw_bo *bo, struct iosys_map *map)
+{
+       struct vmw_private *vmw =
+               container_of(bo->tbo.bdev, struct vmw_private, bdev);
+       void *ptr = NULL;
+       int ret;
+
+       if (bo->tbo.base.import_attach) {
+               ret = dma_buf_vmap(bo->tbo.base.dma_buf, map);
+               if (ret) {
+                       drm_dbg_driver(&vmw->drm,
+                                      "Wasn't able to map external bo!\n");
+                       goto out;
+               }
+               ptr = map->vaddr;
+       } else {
+               ptr = vmw_bo_map_and_cache(bo);
+       }
+
+out:
+       return ptr;
+}
+
+static void unmap_external(struct vmw_bo *bo, struct iosys_map *map)
+{
+       if (bo->tbo.base.import_attach)
+               dma_buf_vunmap(bo->tbo.base.dma_buf, map);
+       else
+               vmw_bo_unmap(bo);
+}
+
+static int vmw_external_bo_copy(struct vmw_bo *dst, u32 dst_offset,
+                               u32 dst_stride, struct vmw_bo *src,
+                               u32 src_offset, u32 src_stride,
+                               u32 width_in_bytes, u32 height,
+                               struct vmw_diff_cpy *diff)
+{
+       struct vmw_private *vmw =
+               container_of(dst->tbo.bdev, struct vmw_private, bdev);
+       size_t dst_size = dst->tbo.resource->size;
+       size_t src_size = src->tbo.resource->size;
+       struct iosys_map dst_map = {0};
+       struct iosys_map src_map = {0};
+       int ret, i;
+       int x_in_bytes;
+       u8 *vsrc;
+       u8 *vdst;
+
+       vsrc = map_external(src, &src_map);
+       if (!vsrc) {
+               drm_dbg_driver(&vmw->drm, "Wasn't able to map src\n");
+               ret = -ENOMEM;
+               goto out;
+       }
+
+       vdst = map_external(dst, &dst_map);
+       if (!vdst) {
+               drm_dbg_driver(&vmw->drm, "Wasn't able to map dst\n");
+               ret = -ENOMEM;
+               goto out;
+       }
+
+       vsrc += src_offset;
+       vdst += dst_offset;
+       if (src_stride == dst_stride) {
+               dst_size -= dst_offset;
+               src_size -= src_offset;
+               memcpy(vdst, vsrc,
+                      min(dst_stride * height, min(dst_size, src_size)));
+       } else {
+               WARN_ON(dst_stride < width_in_bytes);
+               for (i = 0; i < height; ++i) {
+                       memcpy(vdst, vsrc, width_in_bytes);
+                       vsrc += src_stride;
+                       vdst += dst_stride;
+               }
+       }
+
+       x_in_bytes = (dst_offset % dst_stride);
+       diff->rect.x1 =  x_in_bytes / diff->cpp;
+       diff->rect.y1 = ((dst_offset - x_in_bytes) / dst_stride);
+       diff->rect.x2 = diff->rect.x1 + width_in_bytes / diff->cpp;
+       diff->rect.y2 = diff->rect.y1 + height;
+
+       ret = 0;
+out:
+       unmap_external(src, &src_map);
+       unmap_external(dst, &dst_map);
+
+       return ret;
+}
+
 /**
  * vmw_bo_cpu_blit - in-kernel cpu blit.
  *
- * @dst: Destination buffer object.
+ * @vmw_dst: Destination buffer object.
  * @dst_offset: Destination offset of blit start in bytes.
  * @dst_stride: Destination stride in bytes.
- * @src: Source buffer object.
+ * @vmw_src: Source buffer object.
  * @src_offset: Source offset of blit start in bytes.
  * @src_stride: Source stride in bytes.
  * @w: Width of blit.
  * Neither of the buffer objects may be placed in PCI memory
  * (Fixed memory in TTM terminology) when using this function.
  */
-int vmw_bo_cpu_blit(struct ttm_buffer_object *dst,
+int vmw_bo_cpu_blit(struct vmw_bo *vmw_dst,
                    u32 dst_offset, u32 dst_stride,
-                   struct ttm_buffer_object *src,
+                   struct vmw_bo *vmw_src,
                    u32 src_offset, u32 src_stride,
                    u32 w, u32 h,
                    struct vmw_diff_cpy *diff)
 {
+       struct ttm_buffer_object *src = &vmw_src->tbo;
+       struct ttm_buffer_object *dst = &vmw_dst->tbo;
        struct ttm_operation_ctx ctx = {
                .interruptible = false,
                .no_wait_gpu = false
        int ret = 0;
        struct page **dst_pages = NULL;
        struct page **src_pages = NULL;
+       bool src_external = (src->ttm->page_flags & TTM_TT_FLAG_EXTERNAL) != 0;
+       bool dst_external = (dst->ttm->page_flags & TTM_TT_FLAG_EXTERNAL) != 0;
+
+       if (WARN_ON(dst == src))
+               return -EINVAL;
 
        /* Buffer objects need to be either pinned or reserved: */
        if (!(dst->pin_count))
                        return ret;
        }
 
+       if (src_external || dst_external)
+               return vmw_external_bo_copy(vmw_dst, dst_offset, dst_stride,
+                                           vmw_src, src_offset, src_stride,
+                                           w, h, diff);
+
        if (!src->ttm->pages && src->ttm->sg) {
                src_pages = kvmalloc_array(src->ttm->num_pages,
                                           sizeof(struct page *), GFP_KERNEL);
 
                container_of(dirty->unit, typeof(*stdu), base);
        s32 width, height;
        s32 src_pitch, dst_pitch;
-       struct ttm_buffer_object *src_bo, *dst_bo;
+       struct vmw_bo *src_bo, *dst_bo;
        u32 src_offset, dst_offset;
        struct vmw_diff_cpy diff = VMW_CPU_BLIT_DIFF_INITIALIZER(stdu->cpp);
 
 
        /* Assume we are blitting from Guest (bo) to Host (display_srf) */
        src_pitch = stdu->display_srf->metadata.base_size.width * stdu->cpp;
-       src_bo = &stdu->display_srf->res.guest_memory_bo->tbo;
+       src_bo = stdu->display_srf->res.guest_memory_bo;
        src_offset = ddirty->top * src_pitch + ddirty->left * stdu->cpp;
 
        dst_pitch = ddirty->pitch;
-       dst_bo = &ddirty->buf->tbo;
+       dst_bo = ddirty->buf;
        dst_offset = ddirty->fb_top * dst_pitch + ddirty->fb_left * stdu->cpp;
 
        (void) vmw_bo_cpu_blit(dst_bo, dst_offset, dst_pitch,
        struct vmw_diff_cpy diff = VMW_CPU_BLIT_DIFF_INITIALIZER(0);
        struct vmw_stdu_update_gb_image *cmd_img = cmd;
        struct vmw_stdu_update *cmd_update;
-       struct ttm_buffer_object *src_bo, *dst_bo;
+       struct vmw_bo *src_bo, *dst_bo;
        u32 src_offset, dst_offset;
        s32 src_pitch, dst_pitch;
        s32 width, height;
 
        diff.cpp = stdu->cpp;
 
-       dst_bo = &stdu->display_srf->res.guest_memory_bo->tbo;
+       dst_bo = stdu->display_srf->res.guest_memory_bo;
        dst_pitch = stdu->display_srf->metadata.base_size.width * stdu->cpp;
        dst_offset = bb->y1 * dst_pitch + bb->x1 * stdu->cpp;
 
-       src_bo = &vfbbo->buffer->tbo;
+       src_bo = vfbbo->buffer;
        src_pitch = update->vfb->base.pitches[0];
        src_offset = bo_update->fb_top * src_pitch + bo_update->fb_left *
                stdu->cpp;