UPSTREAM: drm/displayid: Iterate over all DisplayID blocks
authorTomas Bzatek <tomas@bzatek.net>
Sun, 1 May 2016 13:02:45 +0000 (15:02 +0200)
committerZheng Yang <zhengyang@rock-chips.com>
Tue, 20 Jun 2017 09:22:50 +0000 (17:22 +0800)
This will iterate over all DisplayID blocks found in the buffer.
Previously only the first block was parsed.

https://bugs.freedesktop.org/show_bug.cgi?id=95207

Change-Id: I952ea2442e8b7c31d8ca882cff8211f008cdd073
Signed-off-by: Tomas Bzatek <tomas@bzatek.net>
Reviewed-by: Jani Nikula <jani.nikula@intel.com>
Signed-off-by: Dave Airlie <airlied@redhat.com>
Signed-off-by: Zheng Yang <zhengyang@rock-chips.com>
(cherry picked from commit 3a4a2ea39f86c581054794c0a727597745f1084b)

drivers/gpu/drm/drm_edid.c

index 36c7d66fb9df1a427c9da9f0ddd4f19414070348..200aa3997bc684169249abddfe0ec8ffcd2cf140 100644 (file)
@@ -4621,19 +4621,25 @@ static int drm_parse_display_id(struct drm_connector *connector,
                return -EINVAL;
        }
 
-       block = (struct displayid_block *)&displayid[idx + 4];
-       DRM_DEBUG_KMS("block id %d, rev %d, len %d\n",
-                     block->tag, block->rev, block->num_bytes);
-
-       switch (block->tag) {
-       case DATA_BLOCK_TILED_DISPLAY:
-               ret = drm_parse_tiled_block(connector, block);
-               if (ret)
-                       return ret;
-               break;
-       default:
-               printk("unknown displayid tag %d\n", block->tag);
-               break;
+       idx += sizeof(struct displayid_hdr);
+       while (block = (struct displayid_block *)&displayid[idx],
+              idx + sizeof(struct displayid_block) <= length &&
+              idx + sizeof(struct displayid_block) + block->num_bytes <= length &&
+              block->num_bytes > 0) {
+               idx += block->num_bytes + sizeof(struct displayid_block);
+               DRM_DEBUG_KMS("block id 0x%x, rev %d, len %d\n",
+                             block->tag, block->rev, block->num_bytes);
+
+               switch (block->tag) {
+               case DATA_BLOCK_TILED_DISPLAY:
+                       ret = drm_parse_tiled_block(connector, block);
+                       if (ret)
+                               return ret;
+                       break;
+               default:
+                       DRM_DEBUG_KMS("found DisplayID tag 0x%x, unhandled\n", block->tag);
+                       break;
+               }
        }
        return 0;
 }