bcm27xx: remove linux 5.10 compatibility
[openwrt/staging/chunkeey.git] / target / linux / bcm27xx / patches-5.10 / 950-0709-drm-Add-GUD-USB-Display-driver.patch
diff --git a/target/linux/bcm27xx/patches-5.10/950-0709-drm-Add-GUD-USB-Display-driver.patch b/target/linux/bcm27xx/patches-5.10/950-0709-drm-Add-GUD-USB-Display-driver.patch
deleted file mode 100644 (file)
index aa10a18..0000000
+++ /dev/null
@@ -1,2676 +0,0 @@
-From 140a4e12d8ec470fe0807931893d3d48ffda46b3 Mon Sep 17 00:00:00 2001
-From: =?UTF-8?q?Noralf=20Tr=C3=B8nnes?= <noralf@tronnes.org>
-Date: Sat, 13 Mar 2021 12:25:45 +0100
-Subject: [PATCH] drm: Add GUD USB Display driver
-MIME-Version: 1.0
-Content-Type: text/plain; charset=UTF-8
-Content-Transfer-Encoding: 8bit
-
-[ Upstream commit 40e1a70b4aedf2859a1829991b48ef0ebe650bf2 ]
-
-This adds a USB display driver with the intention that it can be
-used with future USB interfaced low end displays/adapters. The Linux
-gadget device driver will serve as the canonical device implementation.
-
-The following DRM properties are supported:
-- Plane rotation
-- Connector TV properties
-
-There is also support for backlight brightness exposed as a backlight
-device.
-
-Display modes can be made available to the host driver either as DRM
-display modes or through EDID. If both are present, EDID is just passed
-on to userspace.
-
-Performance is preferred over color depth, so if the device supports
-RGB565, DRM_CAP_DUMB_PREFERRED_DEPTH will return 16.
-
-If the device transfer buffer can't fit an uncompressed framebuffer
-update, the update is split up into parts that do fit.
-
-Optimal user experience is achieved by providing damage reports either by
-setting FB_DAMAGE_CLIPS on pageflips or calling DRM_IOCTL_MODE_DIRTYFB.
-
-LZ4 compression is used if the device supports it.
-
-The driver supports a one bit monochrome transfer format: R1. This is not
-implemented in the gadget driver. It is added in preparation for future
-monochrome e-ink displays.
-
-The driver is MIT licensed to smooth the path for any BSD port of the
-driver.
-
-v2:
-- Use devm_drm_dev_alloc() and drmm_mode_config_init()
-- drm_fbdev_generic_setup: Use preferred_bpp=0, 16 was a copy paste error
-- The drm_backlight_helper is dropped, copy in the code
-- Support protocol version backwards compatibility for device
-
-v3:
-- Use donated Openmoko USB pid
-- Use direct compression from framebuffer when pitch matches, not only on
-  full frames, so split updates can benefit
-- Use __le16 in struct gud_drm_req_get_connector_status
-- Set edid property when the device only provides edid
-- Clear compression fields in struct gud_drm_req_set_buffer
-- Fix protocol version negotiation
-- Remove mode->vrefresh, it's calculated
-
-v4:
-- Drop the status req polling which was a workaround for something that
-  turned out to be a dwc2 udc driver problem
-- Add a flag for the Linux gadget to require a status request on
-  SET operations. Other devices will only get status req on STALL errors
-- Use protocol specific error codes (Peter)
-- Add a flag for devices that want to receive the entire framebuffer on
-  each flush (Lubomir)
-- Retry a failed framebuffer flush
-- If mode has changed wait for worker and clear pending damage before
-  queuing up new damage, fb width/height might have changed
-- Increase error counter on bulk transfer failures
-- Use DRM_MODE_CONNECTOR_USB
-- Handle R1 kmalloc error (Peter)
-- Don't try and replicate the USB get descriptor request standard for the
-  display descriptor (Peter)
-- Make max_buffer_size optional (Peter), drop the pow2 requirement since
-  it's not necessary anymore.
-- Don't pre-alloc a control request buffer, it was only 4k
-- Let gud.h describe the whole protocol explicitly and don't let DRM
-  leak into it (Peter)
-- Drop display mode .hskew and .vscan from the protocol
-- Shorten names: s/GUD_DRM_/GUD_/ s/gud_drm_/gud_/ (Peter)
-- Fix gud_pipe_check() connector picking when switching connector
-- Drop gud_drm_driver_gem_create_object() cached is default now
-- Retrieve USB device from struct drm_device.dev instead of keeping a
-  pointer
-- Honour fb->offsets[0]
-- Fix mode fetching when connector status is forced
-- Check EDID length reported by the device
-- Use drm_do_get_edid() so userspace can overrride EDID
-- Set epoch counter to signal connector status change
-- gud_drm_driver can be const now
-
-v5:
-- GUD_DRM_FORMAT_R1: Use non-human ascii values (Daniel)
-- Change name to: GUD USB Display (Thomas, Simon)
-- Change one __u32 -> __le32 in protocol header
-- Always log fb flush errors, unless the previous one failed
-- Run backlight update in a worker to avoid upsetting lockdep (Daniel)
-- Drop backlight_ops.get_brightness, there's no readback from the device
-  so it doesn't really add anything.
-- Set dma mask, needed by dma-buf importers
-
-v6:
-- Use obj-y in Makefile (Peter)
-- Fix missing le32_to_cpu() when using GUD_DISPLAY_MAGIC (Peter)
-- Set initial brightness on backlight device
-
-v7:
-- LZ4_compress_default() can return zero, check for that
-- Fix memory leak in gud_pipe_check() error path (Peter)
-- Improve debug and error messages (Peter)
-- Don't pass length in protocol structs (Peter)
-- Pass USB interface to gud_usb_control_msg() et al. (Peter)
-- Improve gud_connector_fill_properties() (Peter)
-- Add GUD_PIXEL_FORMAT_RGB111 (Peter)
-- Remove GUD_REQ_SET_VERSION (Peter)
-- Fix DRM_IOCTL_MODE_OBJ_SETPROPERTY and the rotation property
-- Fix dma-buf import (Thomas)
-
-v8:
-- Forgot to filter RGB111 from reaching userspace
-- Handle a device that only returns unknown device properties (Peter)
-- s/GUD_PIXEL_FORMAT_RGB111/GUD_PIXEL_FORMAT_XRGB1111/ (Peter)
-- Fix R1 and XRGB1111 format conversion
-- Add FIXME about Big Endian being broken (Peter, Ilia)
-
-Cc: Lubomir Rintel <lkundrak@v3.sk>
-Acked-by: Daniel Vetter <daniel.vetter@ffwll.ch>
-Reviewed-by: Peter Stuge <peter@stuge.se>
-Tested-by: Peter Stuge <peter@stuge.se>
-Signed-off-by: Noralf Trønnes <noralf@tronnes.org>
-Link: https://patchwork.freedesktop.org/patch/msgid/20210313112545.37527-4-noralf@tronnes.org
-[ backport changes:
-  - Remove include drm_gem_atomic_helper.h
-  - s/drm_gem_simple_display_pipe_prepare_fb/drm_gem_fb_simple_display_pipe_prepare_fb/
-  - Remove const from gud_drm_driver
-  - Change drm_gem_shmem_{vmap,vunmap} signatures
-  - Add gud_gem_create_object() to get cached memory mapping.
-]
-Signed-off-by: Noralf Trønnes <noralf@tronnes.org>
----
- MAINTAINERS                         |   8 +
- drivers/gpu/drm/Kconfig             |   2 +
- drivers/gpu/drm/Makefile            |   1 +
- drivers/gpu/drm/gud/Kconfig         |  14 +
- drivers/gpu/drm/gud/Makefile        |   4 +
- drivers/gpu/drm/gud/gud_connector.c | 729 ++++++++++++++++++++++++++++
- drivers/gpu/drm/gud/gud_drv.c       | 674 +++++++++++++++++++++++++
- drivers/gpu/drm/gud/gud_internal.h  | 154 ++++++
- drivers/gpu/drm/gud/gud_pipe.c      | 551 +++++++++++++++++++++
- include/drm/gud.h                   | 333 +++++++++++++
- 10 files changed, 2470 insertions(+)
- create mode 100644 drivers/gpu/drm/gud/Kconfig
- create mode 100644 drivers/gpu/drm/gud/Makefile
- create mode 100644 drivers/gpu/drm/gud/gud_connector.c
- create mode 100644 drivers/gpu/drm/gud/gud_drv.c
- create mode 100644 drivers/gpu/drm/gud/gud_internal.h
- create mode 100644 drivers/gpu/drm/gud/gud_pipe.c
- create mode 100644 include/drm/gud.h
-
---- a/MAINTAINERS
-+++ b/MAINTAINERS
-@@ -5526,6 +5526,14 @@ S:      Maintained
- F:    Documentation/devicetree/bindings/display/panel/feiyang,fy07024di26a30d.yaml
- F:    drivers/gpu/drm/panel/panel-feiyang-fy07024di26a30d.c
-+DRM DRIVER FOR GENERIC USB DISPLAY
-+M:    Noralf Trønnes <noralf@tronnes.org>
-+S:    Maintained
-+W:    https://github.com/notro/gud/wiki
-+T:    git git://anongit.freedesktop.org/drm/drm-misc
-+F:    drivers/gpu/drm/gud/
-+F:    include/drm/gud.h
-+
- DRM DRIVER FOR GRAIN MEDIA GM12U320 PROJECTORS
- M:    Hans de Goede <hdegoede@redhat.com>
- S:    Maintained
---- a/drivers/gpu/drm/Kconfig
-+++ b/drivers/gpu/drm/Kconfig
-@@ -392,6 +392,8 @@ source "drivers/gpu/drm/tidss/Kconfig"
- source "drivers/gpu/drm/xlnx/Kconfig"
-+source "drivers/gpu/drm/gud/Kconfig"
-+
- # Keep legacy drivers last
- menuconfig DRM_LEGACY
---- a/drivers/gpu/drm/Makefile
-+++ b/drivers/gpu/drm/Makefile
-@@ -124,3 +124,4 @@ obj-$(CONFIG_DRM_ASPEED_GFX) += aspeed/
- obj-$(CONFIG_DRM_MCDE) += mcde/
- obj-$(CONFIG_DRM_TIDSS) += tidss/
- obj-y                 += xlnx/
-+obj-y                 += gud/
---- /dev/null
-+++ b/drivers/gpu/drm/gud/Kconfig
-@@ -0,0 +1,14 @@
-+# SPDX-License-Identifier: GPL-2.0
-+
-+config DRM_GUD
-+      tristate "GUD USB Display"
-+      depends on DRM && USB
-+      select LZ4_COMPRESS
-+      select DRM_KMS_HELPER
-+      select DRM_GEM_SHMEM_HELPER
-+      select BACKLIGHT_CLASS_DEVICE
-+      help
-+        This is a DRM display driver for GUD USB Displays or display
-+        adapters.
-+
-+        If M is selected the module will be called gud.
---- /dev/null
-+++ b/drivers/gpu/drm/gud/Makefile
-@@ -0,0 +1,4 @@
-+# SPDX-License-Identifier: GPL-2.0
-+
-+gud-y                         := gud_drv.o gud_pipe.o gud_connector.o
-+obj-$(CONFIG_DRM_GUD)         += gud.o
---- /dev/null
-+++ b/drivers/gpu/drm/gud/gud_connector.c
-@@ -0,0 +1,729 @@
-+// SPDX-License-Identifier: MIT
-+/*
-+ * Copyright 2020 Noralf Trønnes
-+ */
-+
-+#include <linux/backlight.h>
-+#include <linux/workqueue.h>
-+
-+#include <drm/drm_atomic.h>
-+#include <drm/drm_atomic_state_helper.h>
-+#include <drm/drm_connector.h>
-+#include <drm/drm_drv.h>
-+#include <drm/drm_encoder.h>
-+#include <drm/drm_file.h>
-+#include <drm/drm_modeset_helper_vtables.h>
-+#include <drm/drm_print.h>
-+#include <drm/drm_probe_helper.h>
-+#include <drm/drm_simple_kms_helper.h>
-+#include <drm/gud.h>
-+
-+#include "gud_internal.h"
-+
-+struct gud_connector {
-+      struct drm_connector connector;
-+      struct drm_encoder encoder;
-+      struct backlight_device *backlight;
-+      struct work_struct backlight_work;
-+
-+      /* Supported properties */
-+      u16 *properties;
-+      unsigned int num_properties;
-+
-+      /* Initial gadget tv state if applicable, applied on state reset */
-+      struct drm_tv_connector_state initial_tv_state;
-+
-+      /*
-+       * Initial gadget backlight brightness if applicable, applied on state reset.
-+       * The value -ENODEV is used to signal no backlight.
-+       */
-+      int initial_brightness;
-+};
-+
-+static inline struct gud_connector *to_gud_connector(struct drm_connector *connector)
-+{
-+      return container_of(connector, struct gud_connector, connector);
-+}
-+
-+static void gud_conn_err(struct drm_connector *connector, const char *msg, int ret)
-+{
-+      dev_err(connector->dev->dev, "%s: %s (ret=%d)\n", connector->name, msg, ret);
-+}
-+
-+/*
-+ * Use a worker to avoid taking kms locks inside the backlight lock.
-+ * Other display drivers use backlight within their kms locks.
-+ * This avoids inconsistent locking rules, which would upset lockdep.
-+ */
-+static void gud_connector_backlight_update_status_work(struct work_struct *work)
-+{
-+      struct gud_connector *gconn = container_of(work, struct gud_connector, backlight_work);
-+      struct drm_connector *connector = &gconn->connector;
-+      struct drm_connector_state *connector_state;
-+      struct drm_device *drm = connector->dev;
-+      struct drm_modeset_acquire_ctx ctx;
-+      struct drm_atomic_state *state;
-+      int idx, ret;
-+
-+      if (!drm_dev_enter(drm, &idx))
-+              return;
-+
-+      state = drm_atomic_state_alloc(drm);
-+      if (!state) {
-+              ret = -ENOMEM;
-+              goto exit;
-+      }
-+
-+      drm_modeset_acquire_init(&ctx, 0);
-+      state->acquire_ctx = &ctx;
-+retry:
-+      connector_state = drm_atomic_get_connector_state(state, connector);
-+      if (IS_ERR(connector_state)) {
-+              ret = PTR_ERR(connector_state);
-+              goto out;
-+      }
-+
-+      /* Reuse tv.brightness to avoid having to subclass */
-+      connector_state->tv.brightness = gconn->backlight->props.brightness;
-+
-+      ret = drm_atomic_commit(state);
-+out:
-+      if (ret == -EDEADLK) {
-+              drm_atomic_state_clear(state);
-+              drm_modeset_backoff(&ctx);
-+              goto retry;
-+      }
-+
-+      drm_atomic_state_put(state);
-+
-+      drm_modeset_drop_locks(&ctx);
-+      drm_modeset_acquire_fini(&ctx);
-+exit:
-+      drm_dev_exit(idx);
-+
-+      if (ret)
-+              dev_err(drm->dev, "Failed to update backlight, err=%d\n", ret);
-+}
-+
-+static int gud_connector_backlight_update_status(struct backlight_device *bd)
-+{
-+      struct drm_connector *connector = bl_get_data(bd);
-+      struct gud_connector *gconn = to_gud_connector(connector);
-+
-+      /* The USB timeout is 5 seconds so use system_long_wq for worst case scenario */
-+      queue_work(system_long_wq, &gconn->backlight_work);
-+
-+      return 0;
-+}
-+
-+static const struct backlight_ops gud_connector_backlight_ops = {
-+      .update_status  = gud_connector_backlight_update_status,
-+};
-+
-+static int gud_connector_backlight_register(struct gud_connector *gconn)
-+{
-+      struct drm_connector *connector = &gconn->connector;
-+      struct backlight_device *bd;
-+      const char *name;
-+      const struct backlight_properties props = {
-+              .type = BACKLIGHT_RAW,
-+              .scale = BACKLIGHT_SCALE_NON_LINEAR,
-+              .max_brightness = 100,
-+              .brightness = gconn->initial_brightness,
-+      };
-+
-+      name = kasprintf(GFP_KERNEL, "card%d-%s-backlight",
-+                       connector->dev->primary->index, connector->name);
-+      if (!name)
-+              return -ENOMEM;
-+
-+      bd = backlight_device_register(name, connector->kdev, connector,
-+                                     &gud_connector_backlight_ops, &props);
-+      kfree(name);
-+      if (IS_ERR(bd))
-+              return PTR_ERR(bd);
-+
-+      gconn->backlight = bd;
-+
-+      return 0;
-+}
-+
-+static int gud_connector_detect(struct drm_connector *connector,
-+                              struct drm_modeset_acquire_ctx *ctx, bool force)
-+{
-+      struct gud_device *gdrm = to_gud_device(connector->dev);
-+      int idx, ret;
-+      u8 status;
-+
-+      if (!drm_dev_enter(connector->dev, &idx))
-+              return connector_status_disconnected;
-+
-+      if (force) {
-+              ret = gud_usb_set(gdrm, GUD_REQ_SET_CONNECTOR_FORCE_DETECT,
-+                                connector->index, NULL, 0);
-+              if (ret) {
-+                      ret = connector_status_unknown;
-+                      goto exit;
-+              }
-+      }
-+
-+      ret = gud_usb_get_u8(gdrm, GUD_REQ_GET_CONNECTOR_STATUS, connector->index, &status);
-+      if (ret) {
-+              ret = connector_status_unknown;
-+              goto exit;
-+      }
-+
-+      switch (status & GUD_CONNECTOR_STATUS_CONNECTED_MASK) {
-+      case GUD_CONNECTOR_STATUS_DISCONNECTED:
-+              ret = connector_status_disconnected;
-+              break;
-+      case GUD_CONNECTOR_STATUS_CONNECTED:
-+              ret = connector_status_connected;
-+              break;
-+      default:
-+              ret = connector_status_unknown;
-+              break;
-+      };
-+
-+      if (status & GUD_CONNECTOR_STATUS_CHANGED)
-+              connector->epoch_counter += 1;
-+exit:
-+      drm_dev_exit(idx);
-+
-+      return ret;
-+}
-+
-+struct gud_connector_get_edid_ctx {
-+      void *buf;
-+      size_t len;
-+      bool edid_override;
-+};
-+
-+static int gud_connector_get_edid_block(void *data, u8 *buf, unsigned int block, size_t len)
-+{
-+      struct gud_connector_get_edid_ctx *ctx = data;
-+      size_t start = block * EDID_LENGTH;
-+
-+      ctx->edid_override = false;
-+
-+      if (start + len > ctx->len)
-+              return -1;
-+
-+      memcpy(buf, ctx->buf + start, len);
-+
-+      return 0;
-+}
-+
-+static int gud_connector_get_modes(struct drm_connector *connector)
-+{
-+      struct gud_device *gdrm = to_gud_device(connector->dev);
-+      struct gud_display_mode_req *reqmodes = NULL;
-+      struct gud_connector_get_edid_ctx edid_ctx;
-+      unsigned int i, num_modes = 0;
-+      struct edid *edid = NULL;
-+      int idx, ret;
-+
-+      if (!drm_dev_enter(connector->dev, &idx))
-+              return 0;
-+
-+      edid_ctx.edid_override = true;
-+      edid_ctx.buf = kmalloc(GUD_CONNECTOR_MAX_EDID_LEN, GFP_KERNEL);
-+      if (!edid_ctx.buf)
-+              goto out;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_EDID, connector->index,
-+                        edid_ctx.buf, GUD_CONNECTOR_MAX_EDID_LEN);
-+      if (ret > 0 && ret % EDID_LENGTH) {
-+              gud_conn_err(connector, "Invalid EDID size", ret);
-+      } else if (ret > 0) {
-+              edid_ctx.len = ret;
-+              edid = drm_do_get_edid(connector, gud_connector_get_edid_block, &edid_ctx);
-+      }
-+
-+      kfree(edid_ctx.buf);
-+      drm_connector_update_edid_property(connector, edid);
-+
-+      if (edid && edid_ctx.edid_override)
-+              goto out;
-+
-+      reqmodes = kmalloc_array(GUD_CONNECTOR_MAX_NUM_MODES, sizeof(*reqmodes), GFP_KERNEL);
-+      if (!reqmodes)
-+              goto out;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_MODES, connector->index,
-+                        reqmodes, GUD_CONNECTOR_MAX_NUM_MODES * sizeof(*reqmodes));
-+      if (ret <= 0)
-+              goto out;
-+      if (ret % sizeof(*reqmodes)) {
-+              gud_conn_err(connector, "Invalid display mode array size", ret);
-+              goto out;
-+      }
-+
-+      num_modes = ret / sizeof(*reqmodes);
-+
-+      for (i = 0; i < num_modes; i++) {
-+              struct drm_display_mode *mode;
-+
-+              mode = drm_mode_create(connector->dev);
-+              if (!mode) {
-+                      num_modes = i;
-+                      goto out;
-+              }
-+
-+              gud_to_display_mode(mode, &reqmodes[i]);
-+              drm_mode_probed_add(connector, mode);
-+      }
-+out:
-+      if (!num_modes)
-+              num_modes = drm_add_edid_modes(connector, edid);
-+
-+      kfree(reqmodes);
-+      kfree(edid);
-+      drm_dev_exit(idx);
-+
-+      return num_modes;
-+}
-+
-+static int gud_connector_atomic_check(struct drm_connector *connector,
-+                                    struct drm_atomic_state *state)
-+{
-+      struct drm_connector_state *new_state;
-+      struct drm_crtc_state *new_crtc_state;
-+      struct drm_connector_state *old_state;
-+
-+      new_state = drm_atomic_get_new_connector_state(state, connector);
-+      if (!new_state->crtc)
-+              return 0;
-+
-+      old_state = drm_atomic_get_old_connector_state(state, connector);
-+      new_crtc_state = drm_atomic_get_new_crtc_state(state, new_state->crtc);
-+
-+      if (old_state->tv.margins.left != new_state->tv.margins.left ||
-+          old_state->tv.margins.right != new_state->tv.margins.right ||
-+          old_state->tv.margins.top != new_state->tv.margins.top ||
-+          old_state->tv.margins.bottom != new_state->tv.margins.bottom ||
-+          old_state->tv.mode != new_state->tv.mode ||
-+          old_state->tv.brightness != new_state->tv.brightness ||
-+          old_state->tv.contrast != new_state->tv.contrast ||
-+          old_state->tv.flicker_reduction != new_state->tv.flicker_reduction ||
-+          old_state->tv.overscan != new_state->tv.overscan ||
-+          old_state->tv.saturation != new_state->tv.saturation ||
-+          old_state->tv.hue != new_state->tv.hue)
-+              new_crtc_state->connectors_changed = true;
-+
-+      return 0;
-+}
-+
-+static const struct drm_connector_helper_funcs gud_connector_helper_funcs = {
-+      .detect_ctx = gud_connector_detect,
-+      .get_modes = gud_connector_get_modes,
-+      .atomic_check = gud_connector_atomic_check,
-+};
-+
-+static int gud_connector_late_register(struct drm_connector *connector)
-+{
-+      struct gud_connector *gconn = to_gud_connector(connector);
-+
-+      if (gconn->initial_brightness < 0)
-+              return 0;
-+
-+      return gud_connector_backlight_register(gconn);
-+}
-+
-+static void gud_connector_early_unregister(struct drm_connector *connector)
-+{
-+      struct gud_connector *gconn = to_gud_connector(connector);
-+
-+      backlight_device_unregister(gconn->backlight);
-+      cancel_work_sync(&gconn->backlight_work);
-+}
-+
-+static void gud_connector_destroy(struct drm_connector *connector)
-+{
-+      struct gud_connector *gconn = to_gud_connector(connector);
-+
-+      drm_connector_cleanup(connector);
-+      kfree(gconn->properties);
-+      kfree(gconn);
-+}
-+
-+static void gud_connector_reset(struct drm_connector *connector)
-+{
-+      struct gud_connector *gconn = to_gud_connector(connector);
-+
-+      drm_atomic_helper_connector_reset(connector);
-+      connector->state->tv = gconn->initial_tv_state;
-+      /* Set margins from command line */
-+      drm_atomic_helper_connector_tv_reset(connector);
-+      if (gconn->initial_brightness >= 0)
-+              connector->state->tv.brightness = gconn->initial_brightness;
-+}
-+
-+static const struct drm_connector_funcs gud_connector_funcs = {
-+      .fill_modes = drm_helper_probe_single_connector_modes,
-+      .late_register = gud_connector_late_register,
-+      .early_unregister = gud_connector_early_unregister,
-+      .destroy = gud_connector_destroy,
-+      .reset = gud_connector_reset,
-+      .atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state,
-+      .atomic_destroy_state = drm_atomic_helper_connector_destroy_state,
-+};
-+
-+/*
-+ * The tv.mode property is shared among the connectors and its enum names are
-+ * driver specific. This means that if more than one connector uses tv.mode,
-+ * the enum names has to be the same.
-+ */
-+static int gud_connector_add_tv_mode(struct gud_device *gdrm, struct drm_connector *connector)
-+{
-+      size_t buf_len = GUD_CONNECTOR_TV_MODE_MAX_NUM * GUD_CONNECTOR_TV_MODE_NAME_LEN;
-+      const char *modes[GUD_CONNECTOR_TV_MODE_MAX_NUM];
-+      unsigned int i, num_modes;
-+      char *buf;
-+      int ret;
-+
-+      buf = kmalloc(buf_len, GFP_KERNEL);
-+      if (!buf)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_TV_MODE_VALUES,
-+                        connector->index, buf, buf_len);
-+      if (ret < 0)
-+              goto free;
-+      if (!ret || ret % GUD_CONNECTOR_TV_MODE_NAME_LEN) {
-+              ret = -EIO;
-+              goto free;
-+      }
-+
-+      num_modes = ret / GUD_CONNECTOR_TV_MODE_NAME_LEN;
-+      for (i = 0; i < num_modes; i++)
-+              modes[i] = &buf[i * GUD_CONNECTOR_TV_MODE_NAME_LEN];
-+
-+      ret = drm_mode_create_tv_properties(connector->dev, num_modes, modes);
-+free:
-+      kfree(buf);
-+      if (ret < 0)
-+              gud_conn_err(connector, "Failed to add TV modes", ret);
-+
-+      return ret;
-+}
-+
-+static struct drm_property *
-+gud_connector_property_lookup(struct drm_connector *connector, u16 prop)
-+{
-+      struct drm_mode_config *config = &connector->dev->mode_config;
-+
-+      switch (prop) {
-+      case GUD_PROPERTY_TV_LEFT_MARGIN:
-+              return config->tv_left_margin_property;
-+      case GUD_PROPERTY_TV_RIGHT_MARGIN:
-+              return config->tv_right_margin_property;
-+      case GUD_PROPERTY_TV_TOP_MARGIN:
-+              return config->tv_top_margin_property;
-+      case GUD_PROPERTY_TV_BOTTOM_MARGIN:
-+              return config->tv_bottom_margin_property;
-+      case GUD_PROPERTY_TV_MODE:
-+              return config->tv_mode_property;
-+      case GUD_PROPERTY_TV_BRIGHTNESS:
-+              return config->tv_brightness_property;
-+      case GUD_PROPERTY_TV_CONTRAST:
-+              return config->tv_contrast_property;
-+      case GUD_PROPERTY_TV_FLICKER_REDUCTION:
-+              return config->tv_flicker_reduction_property;
-+      case GUD_PROPERTY_TV_OVERSCAN:
-+              return config->tv_overscan_property;
-+      case GUD_PROPERTY_TV_SATURATION:
-+              return config->tv_saturation_property;
-+      case GUD_PROPERTY_TV_HUE:
-+              return config->tv_hue_property;
-+      default:
-+              return ERR_PTR(-EINVAL);
-+      }
-+}
-+
-+static unsigned int *gud_connector_tv_state_val(u16 prop, struct drm_tv_connector_state *state)
-+{
-+      switch (prop) {
-+      case GUD_PROPERTY_TV_LEFT_MARGIN:
-+              return &state->margins.left;
-+      case GUD_PROPERTY_TV_RIGHT_MARGIN:
-+              return &state->margins.right;
-+      case GUD_PROPERTY_TV_TOP_MARGIN:
-+              return &state->margins.top;
-+      case GUD_PROPERTY_TV_BOTTOM_MARGIN:
-+              return &state->margins.bottom;
-+      case GUD_PROPERTY_TV_MODE:
-+              return &state->mode;
-+      case GUD_PROPERTY_TV_BRIGHTNESS:
-+              return &state->brightness;
-+      case GUD_PROPERTY_TV_CONTRAST:
-+              return &state->contrast;
-+      case GUD_PROPERTY_TV_FLICKER_REDUCTION:
-+              return &state->flicker_reduction;
-+      case GUD_PROPERTY_TV_OVERSCAN:
-+              return &state->overscan;
-+      case GUD_PROPERTY_TV_SATURATION:
-+              return &state->saturation;
-+      case GUD_PROPERTY_TV_HUE:
-+              return &state->hue;
-+      default:
-+              return ERR_PTR(-EINVAL);
-+      }
-+}
-+
-+static int gud_connector_add_properties(struct gud_device *gdrm, struct gud_connector *gconn)
-+{
-+      struct drm_connector *connector = &gconn->connector;
-+      struct drm_device *drm = &gdrm->drm;
-+      struct gud_property_req *properties;
-+      unsigned int i, num_properties;
-+      int ret;
-+
-+      properties = kcalloc(GUD_CONNECTOR_PROPERTIES_MAX_NUM, sizeof(*properties), GFP_KERNEL);
-+      if (!properties)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTOR_PROPERTIES, connector->index,
-+                        properties, GUD_CONNECTOR_PROPERTIES_MAX_NUM * sizeof(*properties));
-+      if (ret <= 0)
-+              goto out;
-+      if (ret % sizeof(*properties)) {
-+              ret = -EIO;
-+              goto out;
-+      }
-+
-+      num_properties = ret / sizeof(*properties);
-+      ret = 0;
-+
-+      gconn->properties = kcalloc(num_properties, sizeof(*gconn->properties), GFP_KERNEL);
-+      if (!gconn->properties) {
-+              ret = -ENOMEM;
-+              goto out;
-+      }
-+
-+      for (i = 0; i < num_properties; i++) {
-+              u16 prop = le16_to_cpu(properties[i].prop);
-+              u64 val = le64_to_cpu(properties[i].val);
-+              struct drm_property *property;
-+              unsigned int *state_val;
-+
-+              drm_dbg(drm, "property: %u = %llu(0x%llx)\n", prop, val, val);
-+
-+              switch (prop) {
-+              case GUD_PROPERTY_TV_LEFT_MARGIN:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_RIGHT_MARGIN:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_TOP_MARGIN:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_BOTTOM_MARGIN:
-+                      ret = drm_mode_create_tv_margin_properties(drm);
-+                      if (ret)
-+                              goto out;
-+                      break;
-+              case GUD_PROPERTY_TV_MODE:
-+                      ret = gud_connector_add_tv_mode(gdrm, connector);
-+                      if (ret)
-+                              goto out;
-+                      break;
-+              case GUD_PROPERTY_TV_BRIGHTNESS:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_CONTRAST:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_FLICKER_REDUCTION:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_OVERSCAN:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_SATURATION:
-+                      fallthrough;
-+              case GUD_PROPERTY_TV_HUE:
-+                      /* This is a no-op if already added. */
-+                      ret = drm_mode_create_tv_properties(drm, 0, NULL);
-+                      if (ret)
-+                              goto out;
-+                      break;
-+              case GUD_PROPERTY_BACKLIGHT_BRIGHTNESS:
-+                      if (val > 100) {
-+                              ret = -EINVAL;
-+                              goto out;
-+                      }
-+                      gconn->initial_brightness = val;
-+                      break;
-+              default:
-+                      /* New ones might show up in future devices, skip those we don't know. */
-+                      drm_dbg(drm, "Ignoring unknown property: %u\n", prop);
-+                      continue;
-+              }
-+
-+              gconn->properties[gconn->num_properties++] = prop;
-+
-+              if (prop == GUD_PROPERTY_BACKLIGHT_BRIGHTNESS)
-+                      continue; /* not a DRM property */
-+
-+              property = gud_connector_property_lookup(connector, prop);
-+              if (WARN_ON(IS_ERR(property)))
-+                      continue;
-+
-+              state_val = gud_connector_tv_state_val(prop, &gconn->initial_tv_state);
-+              if (WARN_ON(IS_ERR(state_val)))
-+                      continue;
-+
-+              *state_val = val;
-+              drm_object_attach_property(&connector->base, property, 0);
-+      }
-+out:
-+      kfree(properties);
-+
-+      return ret;
-+}
-+
-+int gud_connector_fill_properties(struct drm_connector_state *connector_state,
-+                                struct gud_property_req *properties)
-+{
-+      struct gud_connector *gconn = to_gud_connector(connector_state->connector);
-+      unsigned int i;
-+
-+      for (i = 0; i < gconn->num_properties; i++) {
-+              u16 prop = gconn->properties[i];
-+              u64 val;
-+
-+              if (prop == GUD_PROPERTY_BACKLIGHT_BRIGHTNESS) {
-+                      val = connector_state->tv.brightness;
-+              } else {
-+                      unsigned int *state_val;
-+
-+                      state_val = gud_connector_tv_state_val(prop, &connector_state->tv);
-+                      if (WARN_ON_ONCE(IS_ERR(state_val)))
-+                              return PTR_ERR(state_val);
-+
-+                      val = *state_val;
-+              }
-+
-+              properties[i].prop = cpu_to_le16(prop);
-+              properties[i].val = cpu_to_le64(val);
-+      }
-+
-+      return gconn->num_properties;
-+}
-+
-+static int gud_connector_create(struct gud_device *gdrm, unsigned int index,
-+                              struct gud_connector_descriptor_req *desc)
-+{
-+      struct drm_device *drm = &gdrm->drm;
-+      struct gud_connector *gconn;
-+      struct drm_connector *connector;
-+      struct drm_encoder *encoder;
-+      int ret, connector_type;
-+      u32 flags;
-+
-+      gconn = kzalloc(sizeof(*gconn), GFP_KERNEL);
-+      if (!gconn)
-+              return -ENOMEM;
-+
-+      INIT_WORK(&gconn->backlight_work, gud_connector_backlight_update_status_work);
-+      gconn->initial_brightness = -ENODEV;
-+      flags = le32_to_cpu(desc->flags);
-+      connector = &gconn->connector;
-+
-+      drm_dbg(drm, "Connector: index=%u type=%u flags=0x%x\n", index, desc->connector_type, flags);
-+
-+      switch (desc->connector_type) {
-+      case GUD_CONNECTOR_TYPE_PANEL:
-+              connector_type = DRM_MODE_CONNECTOR_USB;
-+              break;
-+      case GUD_CONNECTOR_TYPE_VGA:
-+              connector_type = DRM_MODE_CONNECTOR_VGA;
-+              break;
-+      case GUD_CONNECTOR_TYPE_DVI:
-+              connector_type = DRM_MODE_CONNECTOR_DVID;
-+              break;
-+      case GUD_CONNECTOR_TYPE_COMPOSITE:
-+              connector_type = DRM_MODE_CONNECTOR_Composite;
-+              break;
-+      case GUD_CONNECTOR_TYPE_SVIDEO:
-+              connector_type = DRM_MODE_CONNECTOR_SVIDEO;
-+              break;
-+      case GUD_CONNECTOR_TYPE_COMPONENT:
-+              connector_type = DRM_MODE_CONNECTOR_Component;
-+              break;
-+      case GUD_CONNECTOR_TYPE_DISPLAYPORT:
-+              connector_type = DRM_MODE_CONNECTOR_DisplayPort;
-+              break;
-+      case GUD_CONNECTOR_TYPE_HDMI:
-+              connector_type = DRM_MODE_CONNECTOR_HDMIA;
-+              break;
-+      default: /* future types */
-+              connector_type = DRM_MODE_CONNECTOR_USB;
-+              break;
-+      };
-+
-+      drm_connector_helper_add(connector, &gud_connector_helper_funcs);
-+      ret = drm_connector_init(drm, connector, &gud_connector_funcs, connector_type);
-+      if (ret) {
-+              kfree(connector);
-+              return ret;
-+      }
-+
-+      if (WARN_ON(connector->index != index))
-+              return -EINVAL;
-+
-+      if (flags & GUD_CONNECTOR_FLAGS_POLL_STATUS)
-+              connector->polled = (DRM_CONNECTOR_POLL_CONNECT | DRM_CONNECTOR_POLL_DISCONNECT);
-+      if (flags & GUD_CONNECTOR_FLAGS_INTERLACE)
-+              connector->interlace_allowed = true;
-+      if (flags & GUD_CONNECTOR_FLAGS_DOUBLESCAN)
-+              connector->doublescan_allowed = true;
-+
-+      ret = gud_connector_add_properties(gdrm, gconn);
-+      if (ret) {
-+              gud_conn_err(connector, "Failed to add properties", ret);
-+              return ret;
-+      }
-+
-+      /* The first connector is attached to the existing simple pipe encoder */
-+      if (!connector->index) {
-+              encoder = &gdrm->pipe.encoder;
-+      } else {
-+              encoder = &gconn->encoder;
-+
-+              ret = drm_simple_encoder_init(drm, encoder, DRM_MODE_ENCODER_NONE);
-+              if (ret)
-+                      return ret;
-+
-+              encoder->possible_crtcs = 1;
-+      }
-+
-+      return drm_connector_attach_encoder(connector, encoder);
-+}
-+
-+int gud_get_connectors(struct gud_device *gdrm)
-+{
-+      struct gud_connector_descriptor_req *descs;
-+      unsigned int i, num_connectors;
-+      int ret;
-+
-+      descs = kmalloc_array(GUD_CONNECTORS_MAX_NUM, sizeof(*descs), GFP_KERNEL);
-+      if (!descs)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_CONNECTORS, 0,
-+                        descs, GUD_CONNECTORS_MAX_NUM * sizeof(descs));
-+      if (ret < 0)
-+              goto free;
-+      if (!ret || ret % sizeof(*descs)) {
-+              ret = -EIO;
-+              goto free;
-+      }
-+
-+      num_connectors = ret / sizeof(*descs);
-+
-+      for (i = 0; i < num_connectors; i++) {
-+              ret = gud_connector_create(gdrm, i, &descs[i]);
-+              if (ret)
-+                      goto free;
-+      }
-+free:
-+      kfree(descs);
-+
-+      return ret;
-+}
---- /dev/null
-+++ b/drivers/gpu/drm/gud/gud_drv.c
-@@ -0,0 +1,674 @@
-+// SPDX-License-Identifier: MIT
-+/*
-+ * Copyright 2020 Noralf Trønnes
-+ */
-+
-+#include <linux/dma-buf.h>
-+#include <linux/dma-mapping.h>
-+#include <linux/lz4.h>
-+#include <linux/module.h>
-+#include <linux/platform_device.h>
-+#include <linux/string_helpers.h>
-+#include <linux/usb.h>
-+#include <linux/vmalloc.h>
-+#include <linux/workqueue.h>
-+
-+#include <drm/drm_atomic_helper.h>
-+#include <drm/drm_damage_helper.h>
-+#include <drm/drm_debugfs.h>
-+#include <drm/drm_drv.h>
-+#include <drm/drm_fb_helper.h>
-+#include <drm/drm_fourcc.h>
-+#include <drm/drm_gem_framebuffer_helper.h>
-+#include <drm/drm_gem_shmem_helper.h>
-+#include <drm/drm_managed.h>
-+#include <drm/drm_print.h>
-+#include <drm/drm_probe_helper.h>
-+#include <drm/drm_simple_kms_helper.h>
-+#include <drm/gud.h>
-+
-+#include "gud_internal.h"
-+
-+/* Only used internally */
-+static const struct drm_format_info gud_drm_format_r1 = {
-+      .format = GUD_DRM_FORMAT_R1,
-+      .num_planes = 1,
-+      .char_per_block = { 1, 0, 0 },
-+      .block_w = { 8, 0, 0 },
-+      .block_h = { 1, 0, 0 },
-+      .hsub = 1,
-+      .vsub = 1,
-+};
-+
-+static const struct drm_format_info gud_drm_format_xrgb1111 = {
-+      .format = GUD_DRM_FORMAT_XRGB1111,
-+      .num_planes = 1,
-+      .char_per_block = { 1, 0, 0 },
-+      .block_w = { 2, 0, 0 },
-+      .block_h = { 1, 0, 0 },
-+      .hsub = 1,
-+      .vsub = 1,
-+};
-+
-+static int gud_usb_control_msg(struct usb_interface *intf, bool in,
-+                             u8 request, u16 value, void *buf, size_t len)
-+{
-+      u8 requesttype = USB_TYPE_VENDOR | USB_RECIP_INTERFACE;
-+      u8 ifnum = intf->cur_altsetting->desc.bInterfaceNumber;
-+      struct usb_device *usb = interface_to_usbdev(intf);
-+      unsigned int pipe;
-+
-+      if (len && !buf)
-+              return -EINVAL;
-+
-+      if (in) {
-+              pipe = usb_rcvctrlpipe(usb, 0);
-+              requesttype |= USB_DIR_IN;
-+      } else {
-+              pipe = usb_sndctrlpipe(usb, 0);
-+              requesttype |= USB_DIR_OUT;
-+      }
-+
-+      return usb_control_msg(usb, pipe, request, requesttype, value,
-+                             ifnum, buf, len, USB_CTRL_GET_TIMEOUT);
-+}
-+
-+static int gud_get_display_descriptor(struct usb_interface *intf,
-+                                    struct gud_display_descriptor_req *desc)
-+{
-+      void *buf;
-+      int ret;
-+
-+      buf = kmalloc(sizeof(*desc), GFP_KERNEL);
-+      if (!buf)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_control_msg(intf, true, GUD_REQ_GET_DESCRIPTOR, 0, buf, sizeof(*desc));
-+      memcpy(desc, buf, sizeof(*desc));
-+      kfree(buf);
-+      if (ret < 0)
-+              return ret;
-+      if (ret != sizeof(*desc))
-+              return -EIO;
-+
-+      if (desc->magic != le32_to_cpu(GUD_DISPLAY_MAGIC))
-+              return -ENODATA;
-+
-+      DRM_DEV_DEBUG_DRIVER(&intf->dev,
-+                           "version=%u flags=0x%x compression=0x%x max_buffer_size=%u\n",
-+                           desc->version, le32_to_cpu(desc->flags), desc->compression,
-+                           le32_to_cpu(desc->max_buffer_size));
-+
-+      if (!desc->version || !desc->max_width || !desc->max_height ||
-+          le32_to_cpu(desc->min_width) > le32_to_cpu(desc->max_width) ||
-+          le32_to_cpu(desc->min_height) > le32_to_cpu(desc->max_height))
-+              return -EINVAL;
-+
-+      return 0;
-+}
-+
-+static int gud_status_to_errno(u8 status)
-+{
-+      switch (status) {
-+      case GUD_STATUS_OK:
-+              return 0;
-+      case GUD_STATUS_BUSY:
-+              return -EBUSY;
-+      case GUD_STATUS_REQUEST_NOT_SUPPORTED:
-+              return -EOPNOTSUPP;
-+      case GUD_STATUS_PROTOCOL_ERROR:
-+              return -EPROTO;
-+      case GUD_STATUS_INVALID_PARAMETER:
-+              return -EINVAL;
-+      case GUD_STATUS_ERROR:
-+              return -EREMOTEIO;
-+      default:
-+              return -EREMOTEIO;
-+      }
-+}
-+
-+static int gud_usb_get_status(struct usb_interface *intf)
-+{
-+      int ret, status = -EIO;
-+      u8 *buf;
-+
-+      buf = kmalloc(sizeof(*buf), GFP_KERNEL);
-+      if (!buf)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_control_msg(intf, true, GUD_REQ_GET_STATUS, 0, buf, sizeof(*buf));
-+      if (ret == sizeof(*buf))
-+              status = gud_status_to_errno(*buf);
-+      kfree(buf);
-+
-+      if (ret < 0)
-+              return ret;
-+
-+      return status;
-+}
-+
-+static int gud_usb_transfer(struct gud_device *gdrm, bool in, u8 request, u16 index,
-+                          void *buf, size_t len)
-+{
-+      struct usb_interface *intf = to_usb_interface(gdrm->drm.dev);
-+      int idx, ret;
-+
-+      drm_dbg(&gdrm->drm, "%s: request=0x%x index=%u len=%zu\n",
-+              in ? "get" : "set", request, index, len);
-+
-+      if (!drm_dev_enter(&gdrm->drm, &idx))
-+              return -ENODEV;
-+
-+      mutex_lock(&gdrm->ctrl_lock);
-+
-+      ret = gud_usb_control_msg(intf, in, request, index, buf, len);
-+      if (ret == -EPIPE || ((gdrm->flags & GUD_DISPLAY_FLAG_STATUS_ON_SET) && !in && ret >= 0)) {
-+              int status;
-+
-+              status = gud_usb_get_status(intf);
-+              if (status < 0) {
-+                      ret = status;
-+              } else if (ret < 0) {
-+                      dev_err_once(gdrm->drm.dev,
-+                                   "Unexpected status OK for failed transfer\n");
-+                      ret = -EPIPE;
-+              }
-+      }
-+
-+      if (ret < 0) {
-+              drm_dbg(&gdrm->drm, "ret=%d\n", ret);
-+              gdrm->stats_num_errors++;
-+      }
-+
-+      mutex_unlock(&gdrm->ctrl_lock);
-+      drm_dev_exit(idx);
-+
-+      return ret;
-+}
-+
-+/*
-+ * @buf cannot be allocated on the stack.
-+ * Returns number of bytes received or negative error code on failure.
-+ */
-+int gud_usb_get(struct gud_device *gdrm, u8 request, u16 index, void *buf, size_t max_len)
-+{
-+      return gud_usb_transfer(gdrm, true, request, index, buf, max_len);
-+}
-+
-+/*
-+ * @buf can be allocated on the stack or NULL.
-+ * Returns zero on success or negative error code on failure.
-+ */
-+int gud_usb_set(struct gud_device *gdrm, u8 request, u16 index, void *buf, size_t len)
-+{
-+      void *trbuf = NULL;
-+      int ret;
-+
-+      if (buf && len) {
-+              trbuf = kmemdup(buf, len, GFP_KERNEL);
-+              if (!trbuf)
-+                      return -ENOMEM;
-+      }
-+
-+      ret = gud_usb_transfer(gdrm, false, request, index, trbuf, len);
-+      kfree(trbuf);
-+      if (ret < 0)
-+              return ret;
-+
-+      return ret != len ? -EIO : 0;
-+}
-+
-+/*
-+ * @val can be allocated on the stack.
-+ * Returns zero on success or negative error code on failure.
-+ */
-+int gud_usb_get_u8(struct gud_device *gdrm, u8 request, u16 index, u8 *val)
-+{
-+      u8 *buf;
-+      int ret;
-+
-+      buf = kmalloc(sizeof(*val), GFP_KERNEL);
-+      if (!buf)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_get(gdrm, request, index, buf, sizeof(*val));
-+      *val = *buf;
-+      kfree(buf);
-+      if (ret < 0)
-+              return ret;
-+
-+      return ret != sizeof(*val) ? -EIO : 0;
-+}
-+
-+/* Returns zero on success or negative error code on failure. */
-+int gud_usb_set_u8(struct gud_device *gdrm, u8 request, u8 val)
-+{
-+      return gud_usb_set(gdrm, request, 0, &val, sizeof(val));
-+}
-+
-+static int gud_get_properties(struct gud_device *gdrm)
-+{
-+      struct gud_property_req *properties;
-+      unsigned int i, num_properties;
-+      int ret;
-+
-+      properties = kcalloc(GUD_PROPERTIES_MAX_NUM, sizeof(*properties), GFP_KERNEL);
-+      if (!properties)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_PROPERTIES, 0,
-+                        properties, GUD_PROPERTIES_MAX_NUM * sizeof(*properties));
-+      if (ret <= 0)
-+              goto out;
-+      if (ret % sizeof(*properties)) {
-+              ret = -EIO;
-+              goto out;
-+      }
-+
-+      num_properties = ret / sizeof(*properties);
-+      ret = 0;
-+
-+      gdrm->properties = drmm_kcalloc(&gdrm->drm, num_properties, sizeof(*gdrm->properties),
-+                                      GFP_KERNEL);
-+      if (!gdrm->properties) {
-+              ret = -ENOMEM;
-+              goto out;
-+      }
-+
-+      for (i = 0; i < num_properties; i++) {
-+              u16 prop = le16_to_cpu(properties[i].prop);
-+              u64 val = le64_to_cpu(properties[i].val);
-+
-+              switch (prop) {
-+              case GUD_PROPERTY_ROTATION:
-+                      /*
-+                       * DRM UAPI matches the protocol so use the value directly,
-+                       * but mask out any additions on future devices.
-+                       */
-+                      val &= GUD_ROTATION_MASK;
-+                      ret = drm_plane_create_rotation_property(&gdrm->pipe.plane,
-+                                                               DRM_MODE_ROTATE_0, val);
-+                      break;
-+              default:
-+                      /* New ones might show up in future devices, skip those we don't know. */
-+                      drm_dbg(&gdrm->drm, "Ignoring unknown property: %u\n", prop);
-+                      continue;
-+              }
-+
-+              if (ret)
-+                      goto out;
-+
-+              gdrm->properties[gdrm->num_properties++] = prop;
-+      }
-+out:
-+      kfree(properties);
-+
-+      return ret;
-+}
-+
-+static struct drm_gem_object *gud_gem_create_object(struct drm_device *dev, size_t size)
-+{
-+      struct drm_gem_shmem_object *shmem;
-+
-+      shmem = kzalloc(sizeof(*shmem), GFP_KERNEL);
-+      if (!shmem)
-+              return NULL;
-+
-+      shmem->map_cached = true;
-+
-+      return &shmem->base;
-+}
-+
-+/*
-+ * FIXME: Dma-buf sharing requires DMA support by the importing device.
-+ *        This function is a workaround to make USB devices work as well.
-+ *        See todo.rst for how to fix the issue in the dma-buf framework.
-+ */
-+static struct drm_gem_object *gud_gem_prime_import(struct drm_device *drm, struct dma_buf *dma_buf)
-+{
-+      struct gud_device *gdrm = to_gud_device(drm);
-+
-+      if (!gdrm->dmadev)
-+              return ERR_PTR(-ENODEV);
-+
-+      return drm_gem_prime_import_dev(drm, dma_buf, gdrm->dmadev);
-+}
-+
-+static int gud_stats_debugfs(struct seq_file *m, void *data)
-+{
-+      struct drm_info_node *node = m->private;
-+      struct gud_device *gdrm = to_gud_device(node->minor->dev);
-+      char buf[10];
-+
-+      string_get_size(gdrm->bulk_len, 1, STRING_UNITS_2, buf, sizeof(buf));
-+      seq_printf(m, "Max buffer size: %s\n", buf);
-+      seq_printf(m, "Number of errors:  %u\n", gdrm->stats_num_errors);
-+
-+      seq_puts(m, "Compression:      ");
-+      if (gdrm->compression & GUD_COMPRESSION_LZ4)
-+              seq_puts(m, " lz4");
-+      if (!gdrm->compression)
-+              seq_puts(m, " none");
-+      seq_puts(m, "\n");
-+
-+      if (gdrm->compression) {
-+              u64 remainder;
-+              u64 ratio = div64_u64_rem(gdrm->stats_length, gdrm->stats_actual_length,
-+                                        &remainder);
-+              u64 ratio_frac = div64_u64(remainder * 10, gdrm->stats_actual_length);
-+
-+              seq_printf(m, "Compression ratio: %llu.%llu\n", ratio, ratio_frac);
-+      }
-+
-+      return 0;
-+}
-+
-+static const struct drm_info_list gud_debugfs_list[] = {
-+      { "stats", gud_stats_debugfs, 0, NULL },
-+};
-+
-+static void gud_debugfs_init(struct drm_minor *minor)
-+{
-+      drm_debugfs_create_files(gud_debugfs_list, ARRAY_SIZE(gud_debugfs_list),
-+                               minor->debugfs_root, minor);
-+}
-+
-+static const struct drm_simple_display_pipe_funcs gud_pipe_funcs = {
-+      .check      = gud_pipe_check,
-+      .update     = gud_pipe_update,
-+      .prepare_fb = drm_gem_fb_simple_display_pipe_prepare_fb,
-+};
-+
-+static const struct drm_mode_config_funcs gud_mode_config_funcs = {
-+      .fb_create = drm_gem_fb_create_with_dirty,
-+      .atomic_check = drm_atomic_helper_check,
-+      .atomic_commit = drm_atomic_helper_commit,
-+};
-+
-+static const u64 gud_pipe_modifiers[] = {
-+      DRM_FORMAT_MOD_LINEAR,
-+      DRM_FORMAT_MOD_INVALID
-+};
-+
-+DEFINE_DRM_GEM_FOPS(gud_fops);
-+
-+static struct drm_driver gud_drm_driver = {
-+      .driver_features        = DRIVER_MODESET | DRIVER_GEM | DRIVER_ATOMIC,
-+      .fops                   = &gud_fops,
-+      DRM_GEM_SHMEM_DRIVER_OPS,
-+      .gem_create_object      = gud_gem_create_object,
-+      .gem_prime_import       = gud_gem_prime_import,
-+      .debugfs_init           = gud_debugfs_init,
-+
-+      .name                   = "gud",
-+      .desc                   = "Generic USB Display",
-+      .date                   = "20200422",
-+      .major                  = 1,
-+      .minor                  = 0,
-+};
-+
-+static void gud_free_buffers_and_mutex(struct drm_device *drm, void *unused)
-+{
-+      struct gud_device *gdrm = to_gud_device(drm);
-+
-+      vfree(gdrm->compress_buf);
-+      kfree(gdrm->bulk_buf);
-+      mutex_destroy(&gdrm->ctrl_lock);
-+      mutex_destroy(&gdrm->damage_lock);
-+}
-+
-+static int gud_probe(struct usb_interface *intf, const struct usb_device_id *id)
-+{
-+      const struct drm_format_info *xrgb8888_emulation_format = NULL;
-+      bool rgb565_supported = false, xrgb8888_supported = false;
-+      unsigned int num_formats_dev, num_formats = 0;
-+      struct usb_endpoint_descriptor *bulk_out;
-+      struct gud_display_descriptor_req desc;
-+      struct device *dev = &intf->dev;
-+      size_t max_buffer_size = 0;
-+      struct gud_device *gdrm;
-+      struct drm_device *drm;
-+      u8 *formats_dev;
-+      u32 *formats;
-+      int ret, i;
-+
-+      ret = usb_find_bulk_out_endpoint(intf->cur_altsetting, &bulk_out);
-+      if (ret)
-+              return ret;
-+
-+      ret = gud_get_display_descriptor(intf, &desc);
-+      if (ret) {
-+              DRM_DEV_DEBUG_DRIVER(dev, "Not a display interface: ret=%d\n", ret);
-+              return -ENODEV;
-+      }
-+
-+      if (desc.version > 1) {
-+              dev_err(dev, "Protocol version %u is not supported\n", desc.version);
-+              return -ENODEV;
-+      }
-+
-+      gdrm = devm_drm_dev_alloc(dev, &gud_drm_driver, struct gud_device, drm);
-+      if (IS_ERR(gdrm))
-+              return PTR_ERR(gdrm);
-+
-+      drm = &gdrm->drm;
-+      drm->mode_config.funcs = &gud_mode_config_funcs;
-+      ret = drmm_mode_config_init(drm);
-+      if (ret)
-+              return ret;
-+
-+      gdrm->flags = le32_to_cpu(desc.flags);
-+      gdrm->compression = desc.compression & GUD_COMPRESSION_LZ4;
-+
-+      if (gdrm->flags & GUD_DISPLAY_FLAG_FULL_UPDATE && gdrm->compression)
-+              return -EINVAL;
-+
-+      mutex_init(&gdrm->ctrl_lock);
-+      mutex_init(&gdrm->damage_lock);
-+      INIT_WORK(&gdrm->work, gud_flush_work);
-+      gud_clear_damage(gdrm);
-+
-+      ret = drmm_add_action_or_reset(drm, gud_free_buffers_and_mutex, NULL);
-+      if (ret)
-+              return ret;
-+
-+      drm->mode_config.min_width = le32_to_cpu(desc.min_width);
-+      drm->mode_config.max_width = le32_to_cpu(desc.max_width);
-+      drm->mode_config.min_height = le32_to_cpu(desc.min_height);
-+      drm->mode_config.max_height = le32_to_cpu(desc.max_height);
-+
-+      formats_dev = devm_kmalloc(dev, GUD_FORMATS_MAX_NUM, GFP_KERNEL);
-+      /* Add room for emulated XRGB8888 */
-+      formats = devm_kmalloc_array(dev, GUD_FORMATS_MAX_NUM + 1, sizeof(*formats), GFP_KERNEL);
-+      if (!formats_dev || !formats)
-+              return -ENOMEM;
-+
-+      ret = gud_usb_get(gdrm, GUD_REQ_GET_FORMATS, 0, formats_dev, GUD_FORMATS_MAX_NUM);
-+      if (ret < 0)
-+              return ret;
-+
-+      num_formats_dev = ret;
-+      for (i = 0; i < num_formats_dev; i++) {
-+              const struct drm_format_info *info;
-+              size_t fmt_buf_size;
-+              u32 format;
-+
-+              format = gud_to_fourcc(formats_dev[i]);
-+              if (!format) {
-+                      drm_dbg(drm, "Unsupported format: 0x%02x\n", formats_dev[i]);
-+                      continue;
-+              }
-+
-+              if (format == GUD_DRM_FORMAT_R1)
-+                      info = &gud_drm_format_r1;
-+              else if (format == GUD_DRM_FORMAT_XRGB1111)
-+                      info = &gud_drm_format_xrgb1111;
-+              else
-+                      info = drm_format_info(format);
-+
-+              switch (format) {
-+              case GUD_DRM_FORMAT_R1:
-+                      fallthrough;
-+              case GUD_DRM_FORMAT_XRGB1111:
-+                      if (!xrgb8888_emulation_format)
-+                              xrgb8888_emulation_format = info;
-+                      break;
-+              case DRM_FORMAT_RGB565:
-+                      rgb565_supported = true;
-+                      if (!xrgb8888_emulation_format)
-+                              xrgb8888_emulation_format = info;
-+                      break;
-+              case DRM_FORMAT_XRGB8888:
-+                      xrgb8888_supported = true;
-+                      break;
-+              };
-+
-+              fmt_buf_size = drm_format_info_min_pitch(info, 0, drm->mode_config.max_width) *
-+                             drm->mode_config.max_height;
-+              max_buffer_size = max(max_buffer_size, fmt_buf_size);
-+
-+              if (format == GUD_DRM_FORMAT_R1 || format == GUD_DRM_FORMAT_XRGB1111)
-+                      continue; /* Internal not for userspace */
-+
-+              formats[num_formats++] = format;
-+      }
-+
-+      if (!num_formats && !xrgb8888_emulation_format) {
-+              dev_err(dev, "No supported pixel formats found\n");
-+              return -EINVAL;
-+      }
-+
-+      /* Prefer speed over color depth */
-+      if (rgb565_supported)
-+              drm->mode_config.preferred_depth = 16;
-+
-+      if (!xrgb8888_supported && xrgb8888_emulation_format) {
-+              gdrm->xrgb8888_emulation_format = xrgb8888_emulation_format;
-+              formats[num_formats++] = DRM_FORMAT_XRGB8888;
-+      }
-+
-+      if (desc.max_buffer_size)
-+              max_buffer_size = le32_to_cpu(desc.max_buffer_size);
-+retry:
-+      /*
-+       * Use plain kmalloc here since devm_kmalloc() places struct devres at the beginning
-+       * of the buffer it allocates. This wastes a lot of memory when allocating big buffers.
-+       * Asking for 2M would actually allocate 4M. This would also prevent getting the biggest
-+       * possible buffer potentially leading to split transfers.
-+       */
-+      gdrm->bulk_buf = kmalloc(max_buffer_size, GFP_KERNEL | __GFP_NOWARN);
-+      if (!gdrm->bulk_buf) {
-+              max_buffer_size = roundup_pow_of_two(max_buffer_size) / 2;
-+              if (max_buffer_size < SZ_512K)
-+                      return -ENOMEM;
-+              goto retry;
-+      }
-+
-+      gdrm->bulk_pipe = usb_sndbulkpipe(interface_to_usbdev(intf), usb_endpoint_num(bulk_out));
-+      gdrm->bulk_len = max_buffer_size;
-+
-+      if (gdrm->compression & GUD_COMPRESSION_LZ4) {
-+              gdrm->lz4_comp_mem = devm_kmalloc(dev, LZ4_MEM_COMPRESS, GFP_KERNEL);
-+              if (!gdrm->lz4_comp_mem)
-+                      return -ENOMEM;
-+
-+              gdrm->compress_buf = vmalloc(gdrm->bulk_len);
-+              if (!gdrm->compress_buf)
-+                      return -ENOMEM;
-+      }
-+
-+      ret = drm_simple_display_pipe_init(drm, &gdrm->pipe, &gud_pipe_funcs,
-+                                         formats, num_formats,
-+                                         gud_pipe_modifiers, NULL);
-+      if (ret)
-+              return ret;
-+
-+      devm_kfree(dev, formats);
-+      devm_kfree(dev, formats_dev);
-+
-+      ret = gud_get_properties(gdrm);
-+      if (ret) {
-+              dev_err(dev, "Failed to get properties (error=%d)\n", ret);
-+              return ret;
-+      }
-+
-+      drm_plane_enable_fb_damage_clips(&gdrm->pipe.plane);
-+
-+      ret = gud_get_connectors(gdrm);
-+      if (ret) {
-+              dev_err(dev, "Failed to get connectors (error=%d)\n", ret);
-+              return ret;
-+      }
-+
-+      drm_mode_config_reset(drm);
-+
-+      usb_set_intfdata(intf, gdrm);
-+
-+      gdrm->dmadev = usb_intf_get_dma_device(intf);
-+      if (!gdrm->dmadev)
-+              dev_warn(dev, "buffer sharing not supported");
-+
-+      ret = drm_dev_register(drm, 0);
-+      if (ret) {
-+              put_device(gdrm->dmadev);
-+              return ret;
-+      }
-+
-+      drm_kms_helper_poll_init(drm);
-+
-+      drm_fbdev_generic_setup(drm, 0);
-+
-+      return 0;
-+}
-+
-+static void gud_disconnect(struct usb_interface *interface)
-+{
-+      struct gud_device *gdrm = usb_get_intfdata(interface);
-+      struct drm_device *drm = &gdrm->drm;
-+
-+      drm_dbg(drm, "%s:\n", __func__);
-+
-+      drm_kms_helper_poll_fini(drm);
-+      drm_dev_unplug(drm);
-+      drm_atomic_helper_shutdown(drm);
-+      put_device(gdrm->dmadev);
-+      gdrm->dmadev = NULL;
-+}
-+
-+static int gud_suspend(struct usb_interface *intf, pm_message_t message)
-+{
-+      struct gud_device *gdrm = usb_get_intfdata(intf);
-+
-+      return drm_mode_config_helper_suspend(&gdrm->drm);
-+}
-+
-+static int gud_resume(struct usb_interface *intf)
-+{
-+      struct gud_device *gdrm = usb_get_intfdata(intf);
-+
-+      drm_mode_config_helper_resume(&gdrm->drm);
-+
-+      return 0;
-+}
-+
-+static const struct usb_device_id gud_id_table[] = {
-+      { USB_DEVICE_INTERFACE_CLASS(0x1d50, 0x614d, USB_CLASS_VENDOR_SPEC) },
-+      { }
-+};
-+
-+MODULE_DEVICE_TABLE(usb, gud_id_table);
-+
-+static struct usb_driver gud_usb_driver = {
-+      .name           = "gud",
-+      .probe          = gud_probe,
-+      .disconnect     = gud_disconnect,
-+      .id_table       = gud_id_table,
-+      .suspend        = gud_suspend,
-+      .resume         = gud_resume,
-+      .reset_resume   = gud_resume,
-+};
-+
-+module_usb_driver(gud_usb_driver);
-+
-+MODULE_AUTHOR("Noralf Trønnes");
-+MODULE_LICENSE("Dual MIT/GPL");
---- /dev/null
-+++ b/drivers/gpu/drm/gud/gud_internal.h
-@@ -0,0 +1,154 @@
-+/* SPDX-License-Identifier: MIT */
-+
-+#ifndef __LINUX_GUD_INTERNAL_H
-+#define __LINUX_GUD_INTERNAL_H
-+
-+#include <linux/list.h>
-+#include <linux/mutex.h>
-+#include <linux/usb.h>
-+#include <linux/workqueue.h>
-+#include <uapi/drm/drm_fourcc.h>
-+
-+#include <drm/drm_modes.h>
-+#include <drm/drm_simple_kms_helper.h>
-+
-+struct gud_device {
-+      struct drm_device drm;
-+      struct drm_simple_display_pipe pipe;
-+      struct device *dmadev;
-+      struct work_struct work;
-+      u32 flags;
-+      const struct drm_format_info *xrgb8888_emulation_format;
-+
-+      u16 *properties;
-+      unsigned int num_properties;
-+
-+      unsigned int bulk_pipe;
-+      void *bulk_buf;
-+      size_t bulk_len;
-+
-+      u8 compression;
-+      void *lz4_comp_mem;
-+      void *compress_buf;
-+
-+      u64 stats_length;
-+      u64 stats_actual_length;
-+      unsigned int stats_num_errors;
-+
-+      struct mutex ctrl_lock; /* Serialize get/set and status transfers */
-+
-+      struct mutex damage_lock; /* Protects the following members: */
-+      struct drm_framebuffer *fb;
-+      struct drm_rect damage;
-+      bool prev_flush_failed;
-+};
-+
-+static inline struct gud_device *to_gud_device(struct drm_device *drm)
-+{
-+      return container_of(drm, struct gud_device, drm);
-+}
-+
-+static inline struct usb_device *gud_to_usb_device(struct gud_device *gdrm)
-+{
-+      return interface_to_usbdev(to_usb_interface(gdrm->drm.dev));
-+}
-+
-+int gud_usb_get(struct gud_device *gdrm, u8 request, u16 index, void *buf, size_t len);
-+int gud_usb_set(struct gud_device *gdrm, u8 request, u16 index, void *buf, size_t len);
-+int gud_usb_get_u8(struct gud_device *gdrm, u8 request, u16 index, u8 *val);
-+int gud_usb_set_u8(struct gud_device *gdrm, u8 request, u8 val);
-+
-+void gud_clear_damage(struct gud_device *gdrm);
-+void gud_flush_work(struct work_struct *work);
-+int gud_pipe_check(struct drm_simple_display_pipe *pipe,
-+                 struct drm_plane_state *new_plane_state,
-+                 struct drm_crtc_state *new_crtc_state);
-+void gud_pipe_update(struct drm_simple_display_pipe *pipe,
-+                   struct drm_plane_state *old_state);
-+int gud_connector_fill_properties(struct drm_connector_state *connector_state,
-+                                struct gud_property_req *properties);
-+int gud_get_connectors(struct gud_device *gdrm);
-+
-+/* Driver internal fourcc transfer formats */
-+#define GUD_DRM_FORMAT_R1             0x00000122
-+#define GUD_DRM_FORMAT_XRGB1111               0x03121722
-+
-+static inline u8 gud_from_fourcc(u32 fourcc)
-+{
-+      switch (fourcc) {
-+      case GUD_DRM_FORMAT_R1:
-+              return GUD_PIXEL_FORMAT_R1;
-+      case GUD_DRM_FORMAT_XRGB1111:
-+              return GUD_PIXEL_FORMAT_XRGB1111;
-+      case DRM_FORMAT_RGB565:
-+              return GUD_PIXEL_FORMAT_RGB565;
-+      case DRM_FORMAT_XRGB8888:
-+              return GUD_PIXEL_FORMAT_XRGB8888;
-+      case DRM_FORMAT_ARGB8888:
-+              return GUD_PIXEL_FORMAT_ARGB8888;
-+      };
-+
-+      return 0;
-+}
-+
-+static inline u32 gud_to_fourcc(u8 format)
-+{
-+      switch (format) {
-+      case GUD_PIXEL_FORMAT_R1:
-+              return GUD_DRM_FORMAT_R1;
-+      case GUD_PIXEL_FORMAT_XRGB1111:
-+              return GUD_DRM_FORMAT_XRGB1111;
-+      case GUD_PIXEL_FORMAT_RGB565:
-+              return DRM_FORMAT_RGB565;
-+      case GUD_PIXEL_FORMAT_XRGB8888:
-+              return DRM_FORMAT_XRGB8888;
-+      case GUD_PIXEL_FORMAT_ARGB8888:
-+              return DRM_FORMAT_ARGB8888;
-+      };
-+
-+      return 0;
-+}
-+
-+static inline void gud_from_display_mode(struct gud_display_mode_req *dst,
-+                                       const struct drm_display_mode *src)
-+{
-+      u32 flags = src->flags & GUD_DISPLAY_MODE_FLAG_USER_MASK;
-+
-+      if (src->type & DRM_MODE_TYPE_PREFERRED)
-+              flags |= GUD_DISPLAY_MODE_FLAG_PREFERRED;
-+
-+      dst->clock = cpu_to_le32(src->clock);
-+      dst->hdisplay = cpu_to_le16(src->hdisplay);
-+      dst->hsync_start = cpu_to_le16(src->hsync_start);
-+      dst->hsync_end = cpu_to_le16(src->hsync_end);
-+      dst->htotal = cpu_to_le16(src->htotal);
-+      dst->vdisplay = cpu_to_le16(src->vdisplay);
-+      dst->vsync_start = cpu_to_le16(src->vsync_start);
-+      dst->vsync_end = cpu_to_le16(src->vsync_end);
-+      dst->vtotal = cpu_to_le16(src->vtotal);
-+      dst->flags = cpu_to_le32(flags);
-+}
-+
-+static inline void gud_to_display_mode(struct drm_display_mode *dst,
-+                                     const struct gud_display_mode_req *src)
-+{
-+      u32 flags = le32_to_cpu(src->flags);
-+
-+      memset(dst, 0, sizeof(*dst));
-+      dst->clock = le32_to_cpu(src->clock);
-+      dst->hdisplay = le16_to_cpu(src->hdisplay);
-+      dst->hsync_start = le16_to_cpu(src->hsync_start);
-+      dst->hsync_end = le16_to_cpu(src->hsync_end);
-+      dst->htotal = le16_to_cpu(src->htotal);
-+      dst->vdisplay = le16_to_cpu(src->vdisplay);
-+      dst->vsync_start = le16_to_cpu(src->vsync_start);
-+      dst->vsync_end = le16_to_cpu(src->vsync_end);
-+      dst->vtotal = le16_to_cpu(src->vtotal);
-+      dst->flags = flags & GUD_DISPLAY_MODE_FLAG_USER_MASK;
-+      dst->type = DRM_MODE_TYPE_DRIVER;
-+      if (flags & GUD_DISPLAY_MODE_FLAG_PREFERRED)
-+              dst->type |= DRM_MODE_TYPE_PREFERRED;
-+      drm_mode_set_name(dst);
-+}
-+
-+#endif
---- /dev/null
-+++ b/drivers/gpu/drm/gud/gud_pipe.c
-@@ -0,0 +1,551 @@
-+// SPDX-License-Identifier: MIT
-+/*
-+ * Copyright 2020 Noralf Trønnes
-+ */
-+
-+#include <linux/dma-buf.h>
-+#include <linux/lz4.h>
-+#include <linux/usb.h>
-+#include <linux/workqueue.h>
-+
-+#include <drm/drm_atomic.h>
-+#include <drm/drm_connector.h>
-+#include <drm/drm_damage_helper.h>
-+#include <drm/drm_drv.h>
-+#include <drm/drm_format_helper.h>
-+#include <drm/drm_fourcc.h>
-+#include <drm/drm_framebuffer.h>
-+#include <drm/drm_gem_shmem_helper.h>
-+#include <drm/drm_print.h>
-+#include <drm/drm_rect.h>
-+#include <drm/drm_simple_kms_helper.h>
-+#include <drm/gud.h>
-+
-+#include "gud_internal.h"
-+
-+/*
-+ * FIXME: The driver is probably broken on Big Endian machines.
-+ * See discussion:
-+ * https://lore.kernel.org/dri-devel/CAKb7UvihLX0hgBOP3VBG7O+atwZcUVCPVuBdfmDMpg0NjXe-cQ@mail.gmail.com/
-+ */
-+
-+static bool gud_is_big_endian(void)
-+{
-+#if defined(__BIG_ENDIAN)
-+      return true;
-+#else
-+      return false;
-+#endif
-+}
-+
-+static size_t gud_xrgb8888_to_r124(u8 *dst, const struct drm_format_info *format,
-+                                 void *src, struct drm_framebuffer *fb,
-+                                 struct drm_rect *rect)
-+{
-+      unsigned int block_width = drm_format_info_block_width(format, 0);
-+      unsigned int bits_per_pixel = 8 / block_width;
-+      unsigned int x, y, width, height;
-+      u8 pix, *pix8, *block = dst; /* Assign to silence compiler warning */
-+      size_t len;
-+      void *buf;
-+
-+      WARN_ON_ONCE(format->char_per_block[0] != 1);
-+
-+      /* Start on a byte boundary */
-+      rect->x1 = ALIGN_DOWN(rect->x1, block_width);
-+      width = drm_rect_width(rect);
-+      height = drm_rect_height(rect);
-+      len = drm_format_info_min_pitch(format, 0, width) * height;
-+
-+      buf = kmalloc(width * height, GFP_KERNEL);
-+      if (!buf)
-+              return 0;
-+
-+      drm_fb_xrgb8888_to_gray8(buf, src, fb, rect);
-+      pix8 = buf;
-+
-+      for (y = 0; y < height; y++) {
-+              for (x = 0; x < width; x++) {
-+                      unsigned int pixpos = x % block_width; /* within byte from the left */
-+                      unsigned int pixshift = (block_width - pixpos - 1) * bits_per_pixel;
-+
-+                      if (!pixpos) {
-+                              block = dst++;
-+                              *block = 0;
-+                      }
-+
-+                      pix = (*pix8++) >> (8 - bits_per_pixel);
-+                      *block |= pix << pixshift;
-+              }
-+      }
-+
-+      kfree(buf);
-+
-+      return len;
-+}
-+
-+static size_t gud_xrgb8888_to_color(u8 *dst, const struct drm_format_info *format,
-+                                  void *src, struct drm_framebuffer *fb,
-+                                  struct drm_rect *rect)
-+{
-+      unsigned int block_width = drm_format_info_block_width(format, 0);
-+      unsigned int bits_per_pixel = 8 / block_width;
-+      u8 r, g, b, pix, *block = dst; /* Assign to silence compiler warning */
-+      unsigned int x, y, width;
-+      u32 *pix32;
-+      size_t len;
-+
-+      /* Start on a byte boundary */
-+      rect->x1 = ALIGN_DOWN(rect->x1, block_width);
-+      width = drm_rect_width(rect);
-+      len = drm_format_info_min_pitch(format, 0, width) * drm_rect_height(rect);
-+
-+      for (y = rect->y1; y < rect->y2; y++) {
-+              pix32 = src + (y * fb->pitches[0]);
-+              pix32 += rect->x1;
-+
-+              for (x = 0; x < width; x++) {
-+                      unsigned int pixpos = x % block_width; /* within byte from the left */
-+                      unsigned int pixshift = (block_width - pixpos - 1) * bits_per_pixel;
-+
-+                      if (!pixpos) {
-+                              block = dst++;
-+                              *block = 0;
-+                      }
-+
-+                      r = *pix32 >> 16;
-+                      g = *pix32 >> 8;
-+                      b = *pix32++;
-+
-+                      switch (format->format) {
-+                      case GUD_DRM_FORMAT_XRGB1111:
-+                              pix = ((r >> 7) << 2) | ((g >> 7) << 1) | (b >> 7);
-+                              break;
-+                      default:
-+                              WARN_ON_ONCE(1);
-+                              return len;
-+                      };
-+
-+                      *block |= pix << pixshift;
-+              }
-+      }
-+
-+      return len;
-+}
-+
-+static int gud_prep_flush(struct gud_device *gdrm, struct drm_framebuffer *fb,
-+                        const struct drm_format_info *format, struct drm_rect *rect,
-+                        struct gud_set_buffer_req *req)
-+{
-+      struct dma_buf_attachment *import_attach = fb->obj[0]->import_attach;
-+      u8 compression = gdrm->compression;
-+      void *vmap, *vaddr, *buf;
-+      size_t pitch, len;
-+      int ret = 0;
-+
-+      pitch = drm_format_info_min_pitch(format, 0, drm_rect_width(rect));
-+      len = pitch * drm_rect_height(rect);
-+      if (len > gdrm->bulk_len)
-+              return -E2BIG;
-+
-+      vmap = drm_gem_shmem_vmap(fb->obj[0]);
-+      if (!vmap)
-+              return -ENOMEM;
-+
-+      vaddr = vmap + fb->offsets[0];
-+
-+      if (import_attach) {
-+              ret = dma_buf_begin_cpu_access(import_attach->dmabuf, DMA_FROM_DEVICE);
-+              if (ret)
-+                      goto vunmap;
-+      }
-+retry:
-+      if (compression)
-+              buf = gdrm->compress_buf;
-+      else
-+              buf = gdrm->bulk_buf;
-+
-+      /*
-+       * Imported buffers are assumed to be write-combined and thus uncached
-+       * with slow reads (at least on ARM).
-+       */
-+      if (format != fb->format) {
-+              if (format->format == GUD_DRM_FORMAT_R1) {
-+                      len = gud_xrgb8888_to_r124(buf, format, vaddr, fb, rect);
-+                      if (!len) {
-+                              ret = -ENOMEM;
-+                              goto end_cpu_access;
-+                      }
-+              } else if (format->format == DRM_FORMAT_RGB565) {
-+                      drm_fb_xrgb8888_to_rgb565(buf, vaddr, fb, rect, gud_is_big_endian());
-+              } else {
-+                      len = gud_xrgb8888_to_color(buf, format, vaddr, fb, rect);
-+              }
-+      } else if (gud_is_big_endian() && format->cpp[0] > 1) {
-+              drm_fb_swab(buf, vaddr, fb, rect, !import_attach);
-+      } else if (compression && !import_attach && pitch == fb->pitches[0]) {
-+              /* can compress directly from the framebuffer */
-+              buf = vaddr + rect->y1 * pitch;
-+      } else {
-+              drm_fb_memcpy(buf, vaddr, fb, rect);
-+      }
-+
-+      memset(req, 0, sizeof(*req));
-+      req->x = cpu_to_le32(rect->x1);
-+      req->y = cpu_to_le32(rect->y1);
-+      req->width = cpu_to_le32(drm_rect_width(rect));
-+      req->height = cpu_to_le32(drm_rect_height(rect));
-+      req->length = cpu_to_le32(len);
-+
-+      if (compression & GUD_COMPRESSION_LZ4) {
-+              int complen;
-+
-+              complen = LZ4_compress_default(buf, gdrm->bulk_buf, len, len, gdrm->lz4_comp_mem);
-+              if (complen <= 0) {
-+                      compression = 0;
-+                      goto retry;
-+              }
-+
-+              req->compression = GUD_COMPRESSION_LZ4;
-+              req->compressed_length = cpu_to_le32(complen);
-+      }
-+
-+end_cpu_access:
-+      if (import_attach)
-+              dma_buf_end_cpu_access(import_attach->dmabuf, DMA_FROM_DEVICE);
-+vunmap:
-+      drm_gem_shmem_vunmap(fb->obj[0], vmap);
-+
-+      return ret;
-+}
-+
-+static int gud_flush_rect(struct gud_device *gdrm, struct drm_framebuffer *fb,
-+                        const struct drm_format_info *format, struct drm_rect *rect)
-+{
-+      struct usb_device *usb = gud_to_usb_device(gdrm);
-+      struct gud_set_buffer_req req;
-+      int ret, actual_length;
-+      size_t len, trlen;
-+
-+      drm_dbg(&gdrm->drm, "Flushing [FB:%d] " DRM_RECT_FMT "\n", fb->base.id, DRM_RECT_ARG(rect));
-+
-+      ret = gud_prep_flush(gdrm, fb, format, rect, &req);
-+      if (ret)
-+              return ret;
-+
-+      len = le32_to_cpu(req.length);
-+
-+      if (req.compression)
-+              trlen = le32_to_cpu(req.compressed_length);
-+      else
-+              trlen = len;
-+
-+      gdrm->stats_length += len;
-+      /* Did it wrap around? */
-+      if (gdrm->stats_length <= len && gdrm->stats_actual_length) {
-+              gdrm->stats_length = len;
-+              gdrm->stats_actual_length = 0;
-+      }
-+      gdrm->stats_actual_length += trlen;
-+
-+      if (!(gdrm->flags & GUD_DISPLAY_FLAG_FULL_UPDATE) || gdrm->prev_flush_failed) {
-+              ret = gud_usb_set(gdrm, GUD_REQ_SET_BUFFER, 0, &req, sizeof(req));
-+              if (ret)
-+                      return ret;
-+      }
-+
-+      ret = usb_bulk_msg(usb, gdrm->bulk_pipe, gdrm->bulk_buf, trlen,
-+                         &actual_length, msecs_to_jiffies(3000));
-+      if (!ret && trlen != actual_length)
-+              ret = -EIO;
-+      if (ret)
-+              gdrm->stats_num_errors++;
-+
-+      return ret;
-+}
-+
-+void gud_clear_damage(struct gud_device *gdrm)
-+{
-+      gdrm->damage.x1 = INT_MAX;
-+      gdrm->damage.y1 = INT_MAX;
-+      gdrm->damage.x2 = 0;
-+      gdrm->damage.y2 = 0;
-+}
-+
-+static void gud_add_damage(struct gud_device *gdrm, struct drm_rect *damage)
-+{
-+      gdrm->damage.x1 = min(gdrm->damage.x1, damage->x1);
-+      gdrm->damage.y1 = min(gdrm->damage.y1, damage->y1);
-+      gdrm->damage.x2 = max(gdrm->damage.x2, damage->x2);
-+      gdrm->damage.y2 = max(gdrm->damage.y2, damage->y2);
-+}
-+
-+static void gud_retry_failed_flush(struct gud_device *gdrm, struct drm_framebuffer *fb,
-+                                 struct drm_rect *damage)
-+{
-+      /*
-+       * pipe_update waits for the worker when the display mode is going to change.
-+       * This ensures that the width and height is still the same making it safe to
-+       * add back the damage.
-+       */
-+
-+      mutex_lock(&gdrm->damage_lock);
-+      if (!gdrm->fb) {
-+              drm_framebuffer_get(fb);
-+              gdrm->fb = fb;
-+      }
-+      gud_add_damage(gdrm, damage);
-+      mutex_unlock(&gdrm->damage_lock);
-+
-+      /* Retry only once to avoid a possible storm in case of continues errors. */
-+      if (!gdrm->prev_flush_failed)
-+              queue_work(system_long_wq, &gdrm->work);
-+      gdrm->prev_flush_failed = true;
-+}
-+
-+void gud_flush_work(struct work_struct *work)
-+{
-+      struct gud_device *gdrm = container_of(work, struct gud_device, work);
-+      const struct drm_format_info *format;
-+      struct drm_framebuffer *fb;
-+      struct drm_rect damage;
-+      unsigned int i, lines;
-+      int idx, ret = 0;
-+      size_t pitch;
-+
-+      if (!drm_dev_enter(&gdrm->drm, &idx))
-+              return;
-+
-+      mutex_lock(&gdrm->damage_lock);
-+      fb = gdrm->fb;
-+      gdrm->fb = NULL;
-+      damage = gdrm->damage;
-+      gud_clear_damage(gdrm);
-+      mutex_unlock(&gdrm->damage_lock);
-+
-+      if (!fb)
-+              goto out;
-+
-+      format = fb->format;
-+      if (format->format == DRM_FORMAT_XRGB8888 && gdrm->xrgb8888_emulation_format)
-+              format = gdrm->xrgb8888_emulation_format;
-+
-+      /* Split update if it's too big */
-+      pitch = drm_format_info_min_pitch(format, 0, drm_rect_width(&damage));
-+      lines = drm_rect_height(&damage);
-+
-+      if (gdrm->bulk_len < lines * pitch)
-+              lines = gdrm->bulk_len / pitch;
-+
-+      for (i = 0; i < DIV_ROUND_UP(drm_rect_height(&damage), lines); i++) {
-+              struct drm_rect rect = damage;
-+
-+              rect.y1 += i * lines;
-+              rect.y2 = min_t(u32, rect.y1 + lines, damage.y2);
-+
-+              ret = gud_flush_rect(gdrm, fb, format, &rect);
-+              if (ret) {
-+                      if (ret != -ENODEV && ret != -ECONNRESET &&
-+                          ret != -ESHUTDOWN && ret != -EPROTO) {
-+                              bool prev_flush_failed = gdrm->prev_flush_failed;
-+
-+                              gud_retry_failed_flush(gdrm, fb, &damage);
-+                              if (!prev_flush_failed)
-+                                      dev_err_ratelimited(fb->dev->dev,
-+                                                          "Failed to flush framebuffer: error=%d\n", ret);
-+                      }
-+                      break;
-+              }
-+
-+              gdrm->prev_flush_failed = false;
-+      }
-+
-+      drm_framebuffer_put(fb);
-+out:
-+      drm_dev_exit(idx);
-+}
-+
-+static void gud_fb_queue_damage(struct gud_device *gdrm, struct drm_framebuffer *fb,
-+                              struct drm_rect *damage)
-+{
-+      struct drm_framebuffer *old_fb = NULL;
-+
-+      mutex_lock(&gdrm->damage_lock);
-+
-+      if (fb != gdrm->fb) {
-+              old_fb = gdrm->fb;
-+              drm_framebuffer_get(fb);
-+              gdrm->fb = fb;
-+      }
-+
-+      gud_add_damage(gdrm, damage);
-+
-+      mutex_unlock(&gdrm->damage_lock);
-+
-+      queue_work(system_long_wq, &gdrm->work);
-+
-+      if (old_fb)
-+              drm_framebuffer_put(old_fb);
-+}
-+
-+int gud_pipe_check(struct drm_simple_display_pipe *pipe,
-+                 struct drm_plane_state *new_plane_state,
-+                 struct drm_crtc_state *new_crtc_state)
-+{
-+      struct gud_device *gdrm = to_gud_device(pipe->crtc.dev);
-+      struct drm_plane_state *old_plane_state = pipe->plane.state;
-+      const struct drm_display_mode *mode = &new_crtc_state->mode;
-+      struct drm_atomic_state *state = new_plane_state->state;
-+      struct drm_framebuffer *old_fb = old_plane_state->fb;
-+      struct drm_connector_state *connector_state = NULL;
-+      struct drm_framebuffer *fb = new_plane_state->fb;
-+      const struct drm_format_info *format = fb->format;
-+      struct drm_connector *connector;
-+      unsigned int i, num_properties;
-+      struct gud_state_req *req;
-+      int idx, ret;
-+      size_t len;
-+
-+      if (WARN_ON_ONCE(!fb))
-+              return -EINVAL;
-+
-+      if (old_plane_state->rotation != new_plane_state->rotation)
-+              new_crtc_state->mode_changed = true;
-+
-+      if (old_fb && old_fb->format != format)
-+              new_crtc_state->mode_changed = true;
-+
-+      if (!new_crtc_state->mode_changed && !new_crtc_state->connectors_changed)
-+              return 0;
-+
-+      /* Only one connector is supported */
-+      if (hweight32(new_crtc_state->connector_mask) != 1)
-+              return -EINVAL;
-+
-+      if (format->format == DRM_FORMAT_XRGB8888 && gdrm->xrgb8888_emulation_format)
-+              format = gdrm->xrgb8888_emulation_format;
-+
-+      for_each_new_connector_in_state(state, connector, connector_state, i) {
-+              if (connector_state->crtc)
-+                      break;
-+      }
-+
-+      /*
-+       * DRM_IOCTL_MODE_OBJ_SETPROPERTY on the rotation property will not have
-+       * the connector included in the state.
-+       */
-+      if (!connector_state) {
-+              struct drm_connector_list_iter conn_iter;
-+
-+              drm_connector_list_iter_begin(pipe->crtc.dev, &conn_iter);
-+              drm_for_each_connector_iter(connector, &conn_iter) {
-+                      if (connector->state->crtc) {
-+                              connector_state = connector->state;
-+                              break;
-+                      }
-+              }
-+              drm_connector_list_iter_end(&conn_iter);
-+      }
-+
-+      if (WARN_ON_ONCE(!connector_state))
-+              return -ENOENT;
-+
-+      len = struct_size(req, properties,
-+                        GUD_PROPERTIES_MAX_NUM + GUD_CONNECTOR_PROPERTIES_MAX_NUM);
-+      req = kzalloc(len, GFP_KERNEL);
-+      if (!req)
-+              return -ENOMEM;
-+
-+      gud_from_display_mode(&req->mode, mode);
-+
-+      req->format = gud_from_fourcc(format->format);
-+      if (WARN_ON_ONCE(!req->format)) {
-+              ret = -EINVAL;
-+              goto out;
-+      }
-+
-+      req->connector = drm_connector_index(connector_state->connector);
-+
-+      ret = gud_connector_fill_properties(connector_state, req->properties);
-+      if (ret < 0)
-+              goto out;
-+
-+      num_properties = ret;
-+      for (i = 0; i < gdrm->num_properties; i++) {
-+              u16 prop = gdrm->properties[i];
-+              u64 val;
-+
-+              switch (prop) {
-+              case GUD_PROPERTY_ROTATION:
-+                      /* DRM UAPI matches the protocol so use value directly */
-+                      val = new_plane_state->rotation;
-+                      break;
-+              default:
-+                      WARN_ON_ONCE(1);
-+                      ret = -EINVAL;
-+                      goto out;
-+              }
-+
-+              req->properties[num_properties + i].prop = cpu_to_le16(prop);
-+              req->properties[num_properties + i].val = cpu_to_le64(val);
-+              num_properties++;
-+      }
-+
-+      if (drm_dev_enter(fb->dev, &idx)) {
-+              len = struct_size(req, properties, num_properties);
-+              ret = gud_usb_set(gdrm, GUD_REQ_SET_STATE_CHECK, 0, req, len);
-+              drm_dev_exit(idx);
-+      }  else {
-+              ret = -ENODEV;
-+      }
-+out:
-+      kfree(req);
-+
-+      return ret;
-+}
-+
-+void gud_pipe_update(struct drm_simple_display_pipe *pipe,
-+                   struct drm_plane_state *old_state)
-+{
-+      struct drm_device *drm = pipe->crtc.dev;
-+      struct gud_device *gdrm = to_gud_device(drm);
-+      struct drm_plane_state *state = pipe->plane.state;
-+      struct drm_framebuffer *fb = state->fb;
-+      struct drm_crtc *crtc = &pipe->crtc;
-+      struct drm_rect damage;
-+      int idx;
-+
-+      if (crtc->state->mode_changed || !crtc->state->enable) {
-+              cancel_work_sync(&gdrm->work);
-+              mutex_lock(&gdrm->damage_lock);
-+              if (gdrm->fb) {
-+                      drm_framebuffer_put(gdrm->fb);
-+                      gdrm->fb = NULL;
-+              }
-+              gud_clear_damage(gdrm);
-+              mutex_unlock(&gdrm->damage_lock);
-+      }
-+
-+      if (!drm_dev_enter(drm, &idx))
-+              return;
-+
-+      if (!old_state->fb)
-+              gud_usb_set_u8(gdrm, GUD_REQ_SET_CONTROLLER_ENABLE, 1);
-+
-+      if (fb && (crtc->state->mode_changed || crtc->state->connectors_changed))
-+              gud_usb_set(gdrm, GUD_REQ_SET_STATE_COMMIT, 0, NULL, 0);
-+
-+      if (crtc->state->active_changed)
-+              gud_usb_set_u8(gdrm, GUD_REQ_SET_DISPLAY_ENABLE, crtc->state->active);
-+
-+      if (drm_atomic_helper_damage_merged(old_state, state, &damage)) {
-+              if (gdrm->flags & GUD_DISPLAY_FLAG_FULL_UPDATE)
-+                      drm_rect_init(&damage, 0, 0, fb->width, fb->height);
-+              gud_fb_queue_damage(gdrm, fb, &damage);
-+      }
-+
-+      if (!crtc->state->enable)
-+              gud_usb_set_u8(gdrm, GUD_REQ_SET_CONTROLLER_ENABLE, 0);
-+
-+      drm_dev_exit(idx);
-+}
---- /dev/null
-+++ b/include/drm/gud.h
-@@ -0,0 +1,333 @@
-+/* SPDX-License-Identifier: MIT */
-+/*
-+ * Copyright 2020 Noralf Trønnes
-+ */
-+
-+#ifndef __LINUX_GUD_H
-+#define __LINUX_GUD_H
-+
-+#include <linux/types.h>
-+
-+/*
-+ * struct gud_display_descriptor_req - Display descriptor
-+ * @magic: Magic value GUD_DISPLAY_MAGIC
-+ * @version: Protocol version
-+ * @flags: Flags
-+ *         - STATUS_ON_SET: Always do a status request after a SET request.
-+ *                          This is used by the Linux gadget driver since it has
-+ *                          no way to control the status stage of a control OUT
-+ *                          request that has a payload.
-+ *         - FULL_UPDATE:   Always send the entire framebuffer when flushing changes.
-+ *                          The GUD_REQ_SET_BUFFER request will not be sent
-+ *                          before each bulk transfer, it will only be sent if the
-+ *                          previous bulk transfer had failed. This gives the device
-+ *                          a chance to reset its state machine if needed.
-+ *                          This flag can not be used in combination with compression.
-+ * @compression: Supported compression types
-+ *               - GUD_COMPRESSION_LZ4: LZ4 lossless compression.
-+ * @max_buffer_size: Maximum buffer size the device can handle (optional).
-+ *                   This is useful for devices that don't have a big enough
-+ *                   buffer to decompress the entire framebuffer in one go.
-+ * @min_width: Minimum pixel width the controller can handle
-+ * @max_width: Maximum width
-+ * @min_height: Minimum height
-+ * @max_height: Maximum height
-+ *
-+ * Devices that have only one display mode will have min_width == max_width
-+ * and min_height == max_height.
-+ */
-+struct gud_display_descriptor_req {
-+      __le32 magic;
-+#define GUD_DISPLAY_MAGIC                     0x1d50614d
-+      __u8 version;
-+      __le32 flags;
-+#define GUD_DISPLAY_FLAG_STATUS_ON_SET                BIT(0)
-+#define GUD_DISPLAY_FLAG_FULL_UPDATE          BIT(1)
-+      __u8 compression;
-+#define GUD_COMPRESSION_LZ4                   BIT(0)
-+      __le32 max_buffer_size;
-+      __le32 min_width;
-+      __le32 max_width;
-+      __le32 min_height;
-+      __le32 max_height;
-+} __packed;
-+
-+/*
-+ * struct gud_property_req - Property
-+ * @prop: Property
-+ * @val: Value
-+ */
-+struct gud_property_req {
-+      __le16 prop;
-+      __le64 val;
-+} __packed;
-+
-+/*
-+ * struct gud_display_mode_req - Display mode
-+ * @clock: Pixel clock in kHz
-+ * @hdisplay: Horizontal display size
-+ * @hsync_start: Horizontal sync start
-+ * @hsync_end: Horizontal sync end
-+ * @htotal: Horizontal total size
-+ * @vdisplay: Vertical display size
-+ * @vsync_start: Vertical sync start
-+ * @vsync_end: Vertical sync end
-+ * @vtotal: Vertical total size
-+ * @flags: Bits 0-13 are the same as in the RandR protocol and also what DRM uses.
-+ *         The deprecated bits are reused for internal protocol flags leaving us
-+ *         free to follow DRM for the other bits in the future.
-+ *         - FLAG_PREFERRED: Set on the preferred display mode.
-+ */
-+struct gud_display_mode_req {
-+      __le32 clock;
-+      __le16 hdisplay;
-+      __le16 hsync_start;
-+      __le16 hsync_end;
-+      __le16 htotal;
-+      __le16 vdisplay;
-+      __le16 vsync_start;
-+      __le16 vsync_end;
-+      __le16 vtotal;
-+      __le32 flags;
-+#define GUD_DISPLAY_MODE_FLAG_PHSYNC          BIT(0)
-+#define GUD_DISPLAY_MODE_FLAG_NHSYNC          BIT(1)
-+#define GUD_DISPLAY_MODE_FLAG_PVSYNC          BIT(2)
-+#define GUD_DISPLAY_MODE_FLAG_NVSYNC          BIT(3)
-+#define GUD_DISPLAY_MODE_FLAG_INTERLACE               BIT(4)
-+#define GUD_DISPLAY_MODE_FLAG_DBLSCAN         BIT(5)
-+#define GUD_DISPLAY_MODE_FLAG_CSYNC           BIT(6)
-+#define GUD_DISPLAY_MODE_FLAG_PCSYNC          BIT(7)
-+#define GUD_DISPLAY_MODE_FLAG_NCSYNC          BIT(8)
-+#define GUD_DISPLAY_MODE_FLAG_HSKEW           BIT(9)
-+/* BCast and PixelMultiplex are deprecated */
-+#define GUD_DISPLAY_MODE_FLAG_DBLCLK          BIT(12)
-+#define GUD_DISPLAY_MODE_FLAG_CLKDIV2         BIT(13)
-+#define GUD_DISPLAY_MODE_FLAG_USER_MASK               \
-+              (GUD_DISPLAY_MODE_FLAG_PHSYNC | GUD_DISPLAY_MODE_FLAG_NHSYNC | \
-+              GUD_DISPLAY_MODE_FLAG_PVSYNC | GUD_DISPLAY_MODE_FLAG_NVSYNC | \
-+              GUD_DISPLAY_MODE_FLAG_INTERLACE | GUD_DISPLAY_MODE_FLAG_DBLSCAN | \
-+              GUD_DISPLAY_MODE_FLAG_CSYNC | GUD_DISPLAY_MODE_FLAG_PCSYNC | \
-+              GUD_DISPLAY_MODE_FLAG_NCSYNC | GUD_DISPLAY_MODE_FLAG_HSKEW | \
-+              GUD_DISPLAY_MODE_FLAG_DBLCLK | GUD_DISPLAY_MODE_FLAG_CLKDIV2)
-+/* Internal protocol flags */
-+#define GUD_DISPLAY_MODE_FLAG_PREFERRED               BIT(10)
-+} __packed;
-+
-+/*
-+ * struct gud_connector_descriptor_req - Connector descriptor
-+ * @connector_type: Connector type (GUD_CONNECTOR_TYPE_*).
-+ *                  If the host doesn't support the type it should fall back to PANEL.
-+ * @flags: Flags
-+ *         - POLL_STATUS: Connector status can change (polled every 10 seconds)
-+ *         - INTERLACE: Interlaced modes are supported
-+ *         - DOUBLESCAN: Doublescan modes are supported
-+ */
-+struct gud_connector_descriptor_req {
-+      __u8 connector_type;
-+#define GUD_CONNECTOR_TYPE_PANEL              0
-+#define GUD_CONNECTOR_TYPE_VGA                        1
-+#define GUD_CONNECTOR_TYPE_COMPOSITE          2
-+#define GUD_CONNECTOR_TYPE_SVIDEO             3
-+#define GUD_CONNECTOR_TYPE_COMPONENT          4
-+#define GUD_CONNECTOR_TYPE_DVI                        5
-+#define GUD_CONNECTOR_TYPE_DISPLAYPORT                6
-+#define GUD_CONNECTOR_TYPE_HDMI                       7
-+      __le32 flags;
-+#define GUD_CONNECTOR_FLAGS_POLL_STATUS               BIT(0)
-+#define GUD_CONNECTOR_FLAGS_INTERLACE         BIT(1)
-+#define GUD_CONNECTOR_FLAGS_DOUBLESCAN                BIT(2)
-+} __packed;
-+
-+/*
-+ * struct gud_set_buffer_req - Set buffer transfer info
-+ * @x: X position of rectangle
-+ * @y: Y position
-+ * @width: Pixel width of rectangle
-+ * @height: Pixel height
-+ * @length: Buffer length in bytes
-+ * @compression: Transfer compression
-+ * @compressed_length: Compressed buffer length
-+ *
-+ * This request is issued right before the bulk transfer.
-+ * @x, @y, @width and @height specifies the rectangle where the buffer should be
-+ * placed inside the framebuffer.
-+ */
-+struct gud_set_buffer_req {
-+      __le32 x;
-+      __le32 y;
-+      __le32 width;
-+      __le32 height;
-+      __le32 length;
-+      __u8 compression;
-+      __le32 compressed_length;
-+} __packed;
-+
-+/*
-+ * struct gud_state_req - Display state
-+ * @mode: Display mode
-+ * @format: Pixel format GUD_PIXEL_FORMAT_*
-+ * @connector: Connector index
-+ * @properties: Array of properties
-+ *
-+ * The entire state is transferred each time there's a change.
-+ */
-+struct gud_state_req {
-+      struct gud_display_mode_req mode;
-+      __u8 format;
-+      __u8 connector;
-+      struct gud_property_req properties[];
-+} __packed;
-+
-+/* List of supported connector properties: */
-+
-+/* Margins in pixels to deal with overscan, range 0-100 */
-+#define GUD_PROPERTY_TV_LEFT_MARGIN                   1
-+#define GUD_PROPERTY_TV_RIGHT_MARGIN                  2
-+#define GUD_PROPERTY_TV_TOP_MARGIN                    3
-+#define GUD_PROPERTY_TV_BOTTOM_MARGIN                 4
-+#define GUD_PROPERTY_TV_MODE                          5
-+/* Brightness in percent, range 0-100 */
-+#define GUD_PROPERTY_TV_BRIGHTNESS                    6
-+/* Contrast in percent, range 0-100 */
-+#define GUD_PROPERTY_TV_CONTRAST                      7
-+/* Flicker reduction in percent, range 0-100 */
-+#define GUD_PROPERTY_TV_FLICKER_REDUCTION             8
-+/* Overscan in percent, range 0-100 */
-+#define GUD_PROPERTY_TV_OVERSCAN                      9
-+/* Saturation in percent, range 0-100 */
-+#define GUD_PROPERTY_TV_SATURATION                    10
-+/* Hue in percent, range 0-100 */
-+#define GUD_PROPERTY_TV_HUE                           11
-+
-+/*
-+ * Backlight brightness is in the range 0-100 inclusive. The value represents the human perceptual
-+ * brightness and not a linear PWM value. 0 is minimum brightness which should not turn the
-+ * backlight completely off. The DPMS connector property should be used to control power which will
-+ * trigger a GUD_REQ_SET_DISPLAY_ENABLE request.
-+ *
-+ * This does not map to a DRM property, it is used with the backlight device.
-+ */
-+#define GUD_PROPERTY_BACKLIGHT_BRIGHTNESS             12
-+
-+/* List of supported properties that are not connector propeties: */
-+
-+/*
-+ * Plane rotation. Should return the supported bitmask on
-+ * GUD_REQ_GET_PROPERTIES. GUD_ROTATION_0 is mandatory.
-+ *
-+ * Note: This is not display rotation so 90/270 will need scaling to make it fit (unless squared).
-+ */
-+#define GUD_PROPERTY_ROTATION                         50
-+  #define GUD_ROTATION_0                      BIT(0)
-+  #define GUD_ROTATION_90                     BIT(1)
-+  #define GUD_ROTATION_180                    BIT(2)
-+  #define GUD_ROTATION_270                    BIT(3)
-+  #define GUD_ROTATION_REFLECT_X              BIT(4)
-+  #define GUD_ROTATION_REFLECT_Y              BIT(5)
-+  #define GUD_ROTATION_MASK                   (GUD_ROTATION_0 | GUD_ROTATION_90 | \
-+                                              GUD_ROTATION_180 | GUD_ROTATION_270 | \
-+                                              GUD_ROTATION_REFLECT_X | GUD_ROTATION_REFLECT_Y)
-+
-+/* USB Control requests: */
-+
-+/* Get status from the last GET/SET control request. Value is u8. */
-+#define GUD_REQ_GET_STATUS                            0x00
-+  /* Status values: */
-+  #define GUD_STATUS_OK                               0x00
-+  #define GUD_STATUS_BUSY                     0x01
-+  #define GUD_STATUS_REQUEST_NOT_SUPPORTED    0x02
-+  #define GUD_STATUS_PROTOCOL_ERROR           0x03
-+  #define GUD_STATUS_INVALID_PARAMETER                0x04
-+  #define GUD_STATUS_ERROR                    0x05
-+
-+/* Get display descriptor as a &gud_display_descriptor_req */
-+#define GUD_REQ_GET_DESCRIPTOR                                0x01
-+
-+/* Get supported pixel formats as a byte array of GUD_PIXEL_FORMAT_* */
-+#define GUD_REQ_GET_FORMATS                           0x40
-+  #define GUD_FORMATS_MAX_NUM                 32
-+  /* R1 is a 1-bit monochrome transfer format presented to userspace as XRGB8888 */
-+  #define GUD_PIXEL_FORMAT_R1                 0x01
-+  #define GUD_PIXEL_FORMAT_XRGB1111           0x20
-+  #define GUD_PIXEL_FORMAT_RGB565             0x40
-+  #define GUD_PIXEL_FORMAT_XRGB8888           0x80
-+  #define GUD_PIXEL_FORMAT_ARGB8888           0x81
-+
-+/*
-+ * Get supported properties that are not connector propeties as a &gud_property_req array.
-+ * gud_property_req.val often contains the initial value for the property.
-+ */
-+#define GUD_REQ_GET_PROPERTIES                                0x41
-+  #define GUD_PROPERTIES_MAX_NUM              32
-+
-+/* Connector requests have the connector index passed in the wValue field */
-+
-+/* Get connector descriptors as an array of &gud_connector_descriptor_req */
-+#define GUD_REQ_GET_CONNECTORS                                0x50
-+  #define GUD_CONNECTORS_MAX_NUM              32
-+
-+/*
-+ * Get properties supported by the connector as a &gud_property_req array.
-+ * gud_property_req.val often contains the initial value for the property.
-+ */
-+#define GUD_REQ_GET_CONNECTOR_PROPERTIES              0x51
-+  #define GUD_CONNECTOR_PROPERTIES_MAX_NUM    32
-+
-+/*
-+ * Issued when there's a TV_MODE property present.
-+ * Gets an array of the supported TV_MODE names each entry of length
-+ * GUD_CONNECTOR_TV_MODE_NAME_LEN. Names must be NUL-terminated.
-+ */
-+#define GUD_REQ_GET_CONNECTOR_TV_MODE_VALUES          0x52
-+  #define GUD_CONNECTOR_TV_MODE_NAME_LEN      16
-+  #define GUD_CONNECTOR_TV_MODE_MAX_NUM               16
-+
-+/* When userspace checks connector status, this is issued first, not used for poll requests. */
-+#define GUD_REQ_SET_CONNECTOR_FORCE_DETECT            0x53
-+
-+/*
-+ * Get connector status. Value is u8.
-+ *
-+ * Userspace will get a HOTPLUG uevent if one of the following is true:
-+ * - Connection status has changed since last
-+ * - CHANGED is set
-+ */
-+#define GUD_REQ_GET_CONNECTOR_STATUS                  0x54
-+  #define GUD_CONNECTOR_STATUS_DISCONNECTED   0x00
-+  #define GUD_CONNECTOR_STATUS_CONNECTED      0x01
-+  #define GUD_CONNECTOR_STATUS_UNKNOWN                0x02
-+  #define GUD_CONNECTOR_STATUS_CONNECTED_MASK 0x03
-+  #define GUD_CONNECTOR_STATUS_CHANGED                BIT(7)
-+
-+/*
-+ * Display modes can be fetched as either EDID data or an array of &gud_display_mode_req.
-+ *
-+ * If GUD_REQ_GET_CONNECTOR_MODES returns zero, EDID is used to create display modes.
-+ * If both display modes and EDID are returned, EDID is just passed on to userspace
-+ * in the EDID connector property.
-+ */
-+
-+/* Get &gud_display_mode_req array of supported display modes */
-+#define GUD_REQ_GET_CONNECTOR_MODES                   0x55
-+  #define GUD_CONNECTOR_MAX_NUM_MODES         128
-+
-+/* Get Extended Display Identification Data */
-+#define GUD_REQ_GET_CONNECTOR_EDID                    0x56
-+  #define GUD_CONNECTOR_MAX_EDID_LEN          2048
-+
-+/* Set buffer properties before bulk transfer as &gud_set_buffer_req */
-+#define GUD_REQ_SET_BUFFER                            0x60
-+
-+/* Check display configuration as &gud_state_req */
-+#define GUD_REQ_SET_STATE_CHECK                               0x61
-+
-+/* Apply the previous STATE_CHECK configuration */
-+#define GUD_REQ_SET_STATE_COMMIT                      0x62
-+
-+/* Enable/disable the display controller, value is u8: 0/1 */
-+#define GUD_REQ_SET_CONTROLLER_ENABLE                 0x63
-+
-+/* Enable/disable display/output (DPMS), value is u8: 0/1 */
-+#define GUD_REQ_SET_DISPLAY_ENABLE                    0x64
-+
-+#endif