提交 ab34c226 编写于 作者: C Chris Wilson 提交者: Eric Anholt

drm/i915: Fix up address spaces in slow_kernel_write()

Since we now get_user_pages() outside of the mutex prior to performing
the copy, we kmap() the page inside the copy routine and so need to
perform an ordinary memcpy() and not copy_from_user().
Signed-off-by: NChris Wilson <chris@chris-wilson.co.uk>
Signed-off-by: NEric Anholt <eric@anholt.net>
上级 99a03df5
...@@ -509,25 +509,24 @@ fast_user_write(struct io_mapping *mapping, ...@@ -509,25 +509,24 @@ fast_user_write(struct io_mapping *mapping,
* page faults * page faults
*/ */
static inline int static inline void
slow_kernel_write(struct io_mapping *mapping, slow_kernel_write(struct io_mapping *mapping,
loff_t gtt_base, int gtt_offset, loff_t gtt_base, int gtt_offset,
struct page *user_page, int user_offset, struct page *user_page, int user_offset,
int length) int length)
{ {
char *src_vaddr, *dst_vaddr; char __iomem *dst_vaddr;
unsigned long unwritten; char *src_vaddr;
dst_vaddr = io_mapping_map_wc(mapping, gtt_base);
src_vaddr = kmap(user_page);
dst_vaddr = io_mapping_map_atomic_wc(mapping, gtt_base); memcpy_toio(dst_vaddr + gtt_offset,
src_vaddr = kmap_atomic(user_page, KM_USER1);
unwritten = __copy_from_user_inatomic_nocache(dst_vaddr + gtt_offset,
src_vaddr + user_offset, src_vaddr + user_offset,
length); length);
kunmap_atomic(src_vaddr, KM_USER1);
io_mapping_unmap_atomic(dst_vaddr); kunmap(user_page);
if (unwritten) io_mapping_unmap(dst_vaddr);
return -EFAULT;
return 0;
} }
static inline int static inline int
...@@ -700,19 +699,12 @@ i915_gem_gtt_pwrite_slow(struct drm_device *dev, struct drm_gem_object *obj, ...@@ -700,19 +699,12 @@ i915_gem_gtt_pwrite_slow(struct drm_device *dev, struct drm_gem_object *obj,
if ((data_page_offset + page_length) > PAGE_SIZE) if ((data_page_offset + page_length) > PAGE_SIZE)
page_length = PAGE_SIZE - data_page_offset; page_length = PAGE_SIZE - data_page_offset;
ret = slow_kernel_write(dev_priv->mm.gtt_mapping, slow_kernel_write(dev_priv->mm.gtt_mapping,
gtt_page_base, gtt_page_offset, gtt_page_base, gtt_page_offset,
user_pages[data_page_index], user_pages[data_page_index],
data_page_offset, data_page_offset,
page_length); page_length);
/* If we get a fault while copying data, then (presumably) our
* source page isn't available. Return the error and we'll
* retry in the slow path.
*/
if (ret)
goto out_unpin_object;
remain -= page_length; remain -= page_length;
offset += page_length; offset += page_length;
data_ptr += page_length; data_ptr += page_length;
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册