]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
drm/edid: fix OOB read in drm_parse_tiled_block()
authorXiang Mei <xmei5@asu.edu>
Mon, 15 Jun 2026 18:47:37 +0000 (11:47 -0700)
committerJani Nikula <jani.nikula@intel.com>
Mon, 22 Jun 2026 08:13:27 +0000 (11:13 +0300)
drm_parse_tiled_block() casts the DisplayID block to a
struct displayid_tiled_block and reads the full fixed layout up to
tile->topology_id[7] without checking block->num_bytes. The DisplayID
iterator only validates the declared payload length, so a crafted EDID
can advertise a tiled-display block (tag DATA_BLOCK_TILED_DISPLAY, or
DATA_BLOCK_2_TILED_DISPLAY_TOPOLOGY for v2.0) with a small num_bytes at
the end of a DisplayID extension. The read then runs past the end of the
exact-sized kmemdup()'d EDID allocation, a heap out-of-bounds read.

Reject blocks shorter than the spec's 22-byte tiled payload before
reading the fixed struct, as drm_parse_vesa_mso_data() already does.

  BUG: KASAN: slab-out-of-bounds in drm_edid_connector_update
  Read of size 2 at addr ffff888010077700 by task exploit/147
   dump_stack_lvl (lib/dump_stack.c:94 ...)
   print_report (mm/kasan/report.c:378 ...)
   kasan_report (mm/kasan/report.c:595)
   drm_edid_connector_update (drivers/gpu/drm/drm_edid.c:7581)
   bochs_connector_helper_get_modes (drivers/gpu/drm/tiny/bochs.c:574)
   drm_helper_probe_single_connector_modes (drivers/gpu/drm/drm_probe_helper.c:426)
   status_store (drivers/gpu/drm/drm_sysfs.c:219)
   ...
   vfs_write (fs/read_write.c:595 fs/read_write.c:688)
   ksys_write (fs/read_write.c:740)

Fixes: 40d9b043a89e ("drm/connector: store tile information from displayid (v3)")
Reported-by: Weiming Shi <bestswngs@gmail.com>
Assisted-by: Claude:claude-opus-4-8
Signed-off-by: Xiang Mei <xmei5@asu.edu>
Reviewed-by: Jani Nikula <jani.nikula@intel.com>
Link: https://patch.msgid.link/20260615184737.899892-1-xmei5@asu.edu
Signed-off-by: Jani Nikula <jani.nikula@intel.com>
drivers/gpu/drm/drm_edid.c

index 404208bf23a6b47bacb6dcd0ea9259df007d4ce8..df3c25bac761da0806bd298ecb65e5ce4adea337 100644 (file)
@@ -7575,6 +7575,14 @@ static void drm_parse_tiled_block(struct drm_connector *connector,
        u8 num_v_tile, num_h_tile;
        struct drm_tile_group *tg;
 
+       /* tiled block payload per spec: cap 1 + topo 3 + size 4 + bezel 5 + id 9 = 22 */
+       if (block->num_bytes < 22) {
+               drm_dbg_kms(connector->dev,
+                           "[CONNECTOR:%d:%s] Unexpected tiled block size %u\n",
+                           connector->base.id, connector->name, block->num_bytes);
+               return;
+       }
+
        w = tile->tile_size[0] | tile->tile_size[1] << 8;
        h = tile->tile_size[2] | tile->tile_size[3] << 8;