}
 EXPORT_SYMBOL(drm_fb_swab);
 
-static void drm_fb_xrgb8888_to_rgb332_line(u8 *dbuf, __le32 *sbuf, unsigned int pixels)
+static void drm_fb_xrgb8888_to_rgb332_line(u8 *dbuf, const __le32 *sbuf, unsigned int pixels)
 {
        unsigned int x;
        u32 pix;
 /**
  * drm_fb_xrgb8888_to_rgb332 - Convert XRGB8888 to RGB332 clip buffer
  * @dst: RGB332 destination buffer
+ * @dst_pitch: Number of bytes between two consecutive scanlines within dst
  * @src: XRGB8888 source buffer
  * @fb: DRM framebuffer
  * @clip: Clip rectangle area to copy
  *
  * Drivers can use this function for RGB332 devices that don't natively support XRGB8888.
- *
- * This function does not apply clipping on dst, i.e. the destination is a small buffer
- * containing the clip rect only.
  */
-void drm_fb_xrgb8888_to_rgb332(void *dst, void *src, struct drm_framebuffer *fb,
-                              struct drm_rect *clip)
+void drm_fb_xrgb8888_to_rgb332(void *dst, unsigned int dst_pitch, const void *src,
+                              const struct drm_framebuffer *fb, const struct drm_rect *clip)
 {
        size_t width = drm_rect_width(clip);
        size_t src_len = width * sizeof(u32);
        unsigned int y;
        void *sbuf;
 
+       if (!dst_pitch)
+               dst_pitch = width;
+
        /* Use a buffer to speed up access on buffers with uncached read mapping (i.e. WC) */
        sbuf = kmalloc(src_len, GFP_KERNEL);
        if (!sbuf)
                memcpy(sbuf, src, src_len);
                drm_fb_xrgb8888_to_rgb332_line(dst, sbuf, width);
                src += fb->pitches[0];
-               dst += width;
+               dst += dst_pitch;
        }
 
        kfree(sbuf);
 }
 EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb332);
 
-static void drm_fb_xrgb8888_to_rgb565_line(u16 *dbuf, u32 *sbuf,
+static void drm_fb_xrgb8888_to_rgb565_line(u16 *dbuf, const u32 *sbuf,
                                           unsigned int pixels,
                                           bool swab)
 {
 /**
  * drm_fb_xrgb8888_to_rgb565 - Convert XRGB8888 to RGB565 clip buffer
  * @dst: RGB565 destination buffer
+ * @dst_pitch: Number of bytes between two consecutive scanlines within dst
  * @vaddr: XRGB8888 source buffer
  * @fb: DRM framebuffer
  * @clip: Clip rectangle area to copy
  *
  * Drivers can use this function for RGB565 devices that don't natively
  * support XRGB8888.
- *
- * This function does not apply clipping on dst, i.e. the destination
- * is a small buffer containing the clip rect only.
  */
-void drm_fb_xrgb8888_to_rgb565(void *dst, void *vaddr,
-                              struct drm_framebuffer *fb,
-                              struct drm_rect *clip, bool swab)
+void drm_fb_xrgb8888_to_rgb565(void *dst, unsigned int dst_pitch, const void *vaddr,
+                              const struct drm_framebuffer *fb, const struct drm_rect *clip,
+                              bool swab)
 {
        size_t linepixels = clip->x2 - clip->x1;
        size_t src_len = linepixels * sizeof(u32);
        unsigned y, lines = clip->y2 - clip->y1;
        void *sbuf;
 
+       if (!dst_pitch)
+               dst_pitch = dst_len;
+
        /*
         * The cma memory is write-combined so reads are uncached.
         * Speed up by fetching one line at a time.
                memcpy(sbuf, vaddr, src_len);
                drm_fb_xrgb8888_to_rgb565_line(dst, sbuf, linepixels, swab);
                vaddr += fb->pitches[0];
-               dst += dst_len;
+               dst += dst_pitch;
        }
 
        kfree(sbuf);
 EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb565);
 
 /**
- * drm_fb_xrgb8888_to_rgb565_dstclip - Convert XRGB8888 to RGB565 clip buffer
+ * drm_fb_xrgb8888_to_rgb565_toio - Convert XRGB8888 to RGB565 clip buffer
  * @dst: RGB565 destination buffer (iomem)
- * @dst_pitch: destination buffer pitch
+ * @dst_pitch: Number of bytes between two consecutive scanlines within dst
  * @vaddr: XRGB8888 source buffer
  * @fb: DRM framebuffer
  * @clip: Clip rectangle area to copy
  *
  * Drivers can use this function for RGB565 devices that don't natively
  * support XRGB8888.
- *
- * This function applies clipping on dst, i.e. the destination is a
- * full (iomem) framebuffer but only the clip rect content is copied over.
  */
-void drm_fb_xrgb8888_to_rgb565_dstclip(void __iomem *dst, unsigned int dst_pitch,
-                                      void *vaddr, struct drm_framebuffer *fb,
-                                      struct drm_rect *clip, bool swab)
+void drm_fb_xrgb8888_to_rgb565_toio(void __iomem *dst, unsigned int dst_pitch,
+                                   const void *vaddr, const struct drm_framebuffer *fb,
+                                   const struct drm_rect *clip, bool swab)
 {
        size_t linepixels = clip->x2 - clip->x1;
        size_t dst_len = linepixels * sizeof(u16);
        unsigned y, lines = clip->y2 - clip->y1;
        void *dbuf;
 
+       if (!dst_pitch)
+               dst_pitch = dst_len;
+
        dbuf = kmalloc(dst_len, GFP_KERNEL);
        if (!dbuf)
                return;
 
        vaddr += clip_offset(clip, fb->pitches[0], sizeof(u32));
-       dst += clip_offset(clip, dst_pitch, sizeof(u16));
        for (y = 0; y < lines; y++) {
                drm_fb_xrgb8888_to_rgb565_line(dbuf, vaddr, linepixels, swab);
                memcpy_toio(dst, dbuf, dst_len);
                vaddr += fb->pitches[0];
-               dst += dst_len;
+               dst += dst_pitch;
        }
 
        kfree(dbuf);
 }
-EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb565_dstclip);
+EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb565_toio);
 
-static void drm_fb_xrgb8888_to_rgb888_line(u8 *dbuf, u32 *sbuf,
+static void drm_fb_xrgb8888_to_rgb888_line(u8 *dbuf, const u32 *sbuf,
                                           unsigned int pixels)
 {
        unsigned int x;
 /**
  * drm_fb_xrgb8888_to_rgb888 - Convert XRGB8888 to RGB888 clip buffer
  * @dst: RGB888 destination buffer
+ * @dst_pitch: Number of bytes between two consecutive scanlines within dst
  * @src: XRGB8888 source buffer
  * @fb: DRM framebuffer
  * @clip: Clip rectangle area to copy
  *
  * Drivers can use this function for RGB888 devices that don't natively
  * support XRGB8888.
- *
- * This function does not apply clipping on dst, i.e. the destination
- * is a small buffer containing the clip rect only.
  */
-void drm_fb_xrgb8888_to_rgb888(void *dst, void *src, struct drm_framebuffer *fb,
-                              struct drm_rect *clip)
+void drm_fb_xrgb8888_to_rgb888(void *dst, unsigned int dst_pitch, const void *src,
+                              const struct drm_framebuffer *fb, const struct drm_rect *clip)
 {
        size_t width = drm_rect_width(clip);
        size_t src_len = width * sizeof(u32);
        unsigned int y;
        void *sbuf;
 
+       if (!dst_pitch)
+               dst_pitch = width * 3;
+
        /* Use a buffer to speed up access on buffers with uncached read mapping (i.e. WC) */
        sbuf = kmalloc(src_len, GFP_KERNEL);
        if (!sbuf)
                memcpy(sbuf, src, src_len);
                drm_fb_xrgb8888_to_rgb888_line(dst, sbuf, width);
                src += fb->pitches[0];
-               dst += width * 3;
+               dst += dst_pitch;
        }
 
        kfree(sbuf);
 EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb888);
 
 /**
- * drm_fb_xrgb8888_to_rgb888_dstclip - Convert XRGB8888 to RGB888 clip buffer
+ * drm_fb_xrgb8888_to_rgb888_toio - Convert XRGB8888 to RGB888 clip buffer
  * @dst: RGB565 destination buffer (iomem)
- * @dst_pitch: destination buffer pitch
+ * @dst_pitch: Number of bytes between two consecutive scanlines within dst
  * @vaddr: XRGB8888 source buffer
  * @fb: DRM framebuffer
  * @clip: Clip rectangle area to copy
  *
  * Drivers can use this function for RGB888 devices that don't natively
  * support XRGB8888.
- *
- * This function applies clipping on dst, i.e. the destination is a
- * full (iomem) framebuffer but only the clip rect content is copied over.
  */
-void drm_fb_xrgb8888_to_rgb888_dstclip(void __iomem *dst, unsigned int dst_pitch,
-                                      void *vaddr, struct drm_framebuffer *fb,
-                                      struct drm_rect *clip)
+void drm_fb_xrgb8888_to_rgb888_toio(void __iomem *dst, unsigned int dst_pitch,
+                                   const void *vaddr, const struct drm_framebuffer *fb,
+                                   const struct drm_rect *clip)
 {
        size_t linepixels = clip->x2 - clip->x1;
        size_t dst_len = linepixels * 3;
        unsigned y, lines = clip->y2 - clip->y1;
        void *dbuf;
 
+       if (!dst_pitch)
+               dst_pitch = dst_len;
+
        dbuf = kmalloc(dst_len, GFP_KERNEL);
        if (!dbuf)
                return;
 
        vaddr += clip_offset(clip, fb->pitches[0], sizeof(u32));
-       dst += clip_offset(clip, dst_pitch, sizeof(u16));
        for (y = 0; y < lines; y++) {
                drm_fb_xrgb8888_to_rgb888_line(dbuf, vaddr, linepixels);
                memcpy_toio(dst, dbuf, dst_len);
                vaddr += fb->pitches[0];
-               dst += dst_len;
+               dst += dst_pitch;
        }
 
        kfree(dbuf);
 }
-EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb888_dstclip);
+EXPORT_SYMBOL(drm_fb_xrgb8888_to_rgb888_toio);
 
 /**
  * drm_fb_xrgb8888_to_gray8 - Convert XRGB8888 to grayscale
  * @dst: 8-bit grayscale destination buffer
+ * @dst_pitch: Number of bytes between two consecutive scanlines within dst
  * @vaddr: XRGB8888 source buffer
  * @fb: DRM framebuffer
  * @clip: Clip rectangle area to copy
  *
  * ITU BT.601 is used for the RGB -> luma (brightness) conversion.
  */
-void drm_fb_xrgb8888_to_gray8(u8 *dst, void *vaddr, struct drm_framebuffer *fb,
-                              struct drm_rect *clip)
+void drm_fb_xrgb8888_to_gray8(void *dst, unsigned int dst_pitch, const void *vaddr,
+                             const struct drm_framebuffer *fb, const struct drm_rect *clip)
 {
        unsigned int len = (clip->x2 - clip->x1) * sizeof(u32);
        unsigned int x, y;
        void *buf;
-       u32 *src;
+       u8 *dst8;
+       u32 *src32;
 
        if (WARN_ON(fb->format->format != DRM_FORMAT_XRGB8888))
                return;
+
+       if (!dst_pitch)
+               dst_pitch = drm_rect_width(clip);
+
        /*
         * The cma memory is write-combined so reads are uncached.
         * Speed up by fetching one line at a time.
        if (!buf)
                return;
 
+       vaddr += clip_offset(clip, fb->pitches[0], sizeof(u32));
        for (y = clip->y1; y < clip->y2; y++) {
-               src = vaddr + (y * fb->pitches[0]);
-               src += clip->x1;
-               memcpy(buf, src, len);
-               src = buf;
+               dst8 = dst;
+               src32 = memcpy(buf, vaddr, len);
                for (x = clip->x1; x < clip->x2; x++) {
-                       u8 r = (*src & 0x00ff0000) >> 16;
-                       u8 g = (*src & 0x0000ff00) >> 8;
-                       u8 b =  *src & 0x000000ff;
+                       u8 r = (*src32 & 0x00ff0000) >> 16;
+                       u8 g = (*src32 & 0x0000ff00) >> 8;
+                       u8 b =  *src32 & 0x000000ff;
 
                        /* ITU BT.601: Y = 0.299 R + 0.587 G + 0.114 B */
-                       *dst++ = (3 * r + 6 * g + b) / 10;
-                       src++;
+                       *dst8++ = (3 * r + 6 * g + b) / 10;
+                       src32++;
                }
+
+               vaddr += fb->pitches[0];
+               dst += dst_pitch;
        }
 
        kfree(buf);
 
        } else if (dst_format == DRM_FORMAT_RGB565) {
                if (fb_format == DRM_FORMAT_XRGB8888) {
-                       drm_fb_xrgb8888_to_rgb565_dstclip(dst, dst_pitch,
-                                                         vmap, fb, clip,
-                                                         false);
+                       dst += clip_offset(clip, dst_pitch, fb->format->cpp[0]);
+                       drm_fb_xrgb8888_to_rgb565_toio(dst, dst_pitch, vmap, fb, clip,
+                                                      false);
                        return 0;
                }
        } else if (dst_format == DRM_FORMAT_RGB888) {
                if (fb_format == DRM_FORMAT_XRGB8888) {
-                       drm_fb_xrgb8888_to_rgb888_dstclip(dst, dst_pitch,
-                                                         vmap, fb, clip);
+                       dst += clip_offset(clip, dst_pitch, fb->format->cpp[0]);
+                       drm_fb_xrgb8888_to_rgb888_toio(dst, dst_pitch, vmap, fb, clip);
                        return 0;
                }
        }
 
 void drm_fb_swab(void *dst, unsigned int dst_pitch, const void *src,
                 const struct drm_framebuffer *fb, const struct drm_rect *clip,
                 bool cached);
-void drm_fb_xrgb8888_to_rgb332(void *dst, void *vaddr, struct drm_framebuffer *fb,
-                              struct drm_rect *clip);
-void drm_fb_xrgb8888_to_rgb565(void *dst, void *vaddr,
-                              struct drm_framebuffer *fb,
-                              struct drm_rect *clip, bool swab);
-void drm_fb_xrgb8888_to_rgb565_dstclip(void __iomem *dst, unsigned int dst_pitch,
-                                      void *vaddr, struct drm_framebuffer *fb,
-                                      struct drm_rect *clip, bool swab);
-void drm_fb_xrgb8888_to_rgb888(void *dst, void *src, struct drm_framebuffer *fb,
-                              struct drm_rect *clip);
-void drm_fb_xrgb8888_to_rgb888_dstclip(void __iomem *dst, unsigned int dst_pitch,
-                                      void *vaddr, struct drm_framebuffer *fb,
-                                      struct drm_rect *clip);
-void drm_fb_xrgb8888_to_gray8(u8 *dst, void *vaddr, struct drm_framebuffer *fb,
-                             struct drm_rect *clip);
+void drm_fb_xrgb8888_to_rgb332(void *dst, unsigned int dst_pitch, const void *vaddr,
+                              const struct drm_framebuffer *fb, const struct drm_rect *clip);
+void drm_fb_xrgb8888_to_rgb565(void *dst, unsigned int dst_pitch, const void *vaddr,
+                              const struct drm_framebuffer *fb, const struct drm_rect *clip,
+                              bool swab);
+void drm_fb_xrgb8888_to_rgb565_toio(void __iomem *dst, unsigned int dst_pitch,
+                                   const void *vaddr, const struct drm_framebuffer *fb,
+                                   const struct drm_rect *clip, bool swab);
+void drm_fb_xrgb8888_to_rgb888(void *dst, unsigned int dst_pitch, const void *src,
+                              const struct drm_framebuffer *fb, const struct drm_rect *clip);
+void drm_fb_xrgb8888_to_rgb888_toio(void __iomem *dst, unsigned int dst_pitch,
+                                   const void *vaddr, const struct drm_framebuffer *fb,
+                                   const struct drm_rect *clip);
+void drm_fb_xrgb8888_to_gray8(void *dst, unsigned int dst_pitch, const void *vaddr,
+                             const struct drm_framebuffer *fb, const struct drm_rect *clip);
 
 int drm_fb_blit_rect_dstclip(void __iomem *dst, unsigned int dst_pitch,
                             uint32_t dst_format, void *vmap,