diff options
author | Sebastian Reichel <sebastian.reichel@collabora.com> | 2018-11-21 11:09:14 -0500 |
---|---|---|
committer | Tomi Valkeinen <tomi.valkeinen@ti.com> | 2018-12-03 07:46:30 -0500 |
commit | 3c613a3bddd322c87677604d81e267fee22c6f14 (patch) | |
tree | 3e33feb6471f7e66d1e49d9da797c87a4ca76167 /drivers/gpu/drm/omapdrm/omap_encoder.c | |
parent | 0a02d495531e0bbe32c3f7361232ba61b981199a (diff) |
drm/omap: fix incorrect union usage
The DSI encoder sets dssdev->ops->dsi.set_config, which is stored at the
same offset as dssdev->ops->hdmi.set_hdmi_mode. The code in omap_encoder
only checks if dssdev->ops->hdmi.set_hdmi_mode is NULL. Due to the way
union works, it won't be NULL if dsi.set_config is set. This means
dsi_set_config will be called with config=hdmi_mode=false=NULL parameter
resulting in a NULL dereference. Also the dereference happens while
console is locked, so kernel hangs without any debug output without
"fb.lockless_register_fb=1" parameter.
This restructures the code, so that the HDMI mode is only configured
for HDMI output types.
Fixes: 83910ad3f51fb ("drm/omap: Move most omap_dss_driver operations to omap_dss_device_ops")
Signed-off-by: Sebastian Reichel <sebastian.reichel@collabora.com>
Tested-by: Tony Lindgren <tony@atomide.com>
[tomi.valkeinen@ti.com: dropped the safeguard]
Signed-off-by: Tomi Valkeinen <tomi.valkeinen@ti.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20181121160916.22017-5-sebastian.reichel@collabora.com
Diffstat (limited to 'drivers/gpu/drm/omapdrm/omap_encoder.c')
-rw-r--r-- | drivers/gpu/drm/omapdrm/omap_encoder.c | 58 |
1 files changed, 33 insertions, 25 deletions
diff --git a/drivers/gpu/drm/omapdrm/omap_encoder.c b/drivers/gpu/drm/omapdrm/omap_encoder.c index 452e625f6ce3..933ebc9f9faa 100644 --- a/drivers/gpu/drm/omapdrm/omap_encoder.c +++ b/drivers/gpu/drm/omapdrm/omap_encoder.c | |||
@@ -52,17 +52,44 @@ static const struct drm_encoder_funcs omap_encoder_funcs = { | |||
52 | .destroy = omap_encoder_destroy, | 52 | .destroy = omap_encoder_destroy, |
53 | }; | 53 | }; |
54 | 54 | ||
55 | static void omap_encoder_hdmi_mode_set(struct drm_encoder *encoder, | ||
56 | struct drm_display_mode *adjusted_mode) | ||
57 | { | ||
58 | struct drm_device *dev = encoder->dev; | ||
59 | struct omap_encoder *omap_encoder = to_omap_encoder(encoder); | ||
60 | struct omap_dss_device *dssdev = omap_encoder->output; | ||
61 | struct drm_connector *connector; | ||
62 | bool hdmi_mode; | ||
63 | |||
64 | hdmi_mode = false; | ||
65 | list_for_each_entry(connector, &dev->mode_config.connector_list, head) { | ||
66 | if (connector->encoder == encoder) { | ||
67 | hdmi_mode = omap_connector_get_hdmi_mode(connector); | ||
68 | break; | ||
69 | } | ||
70 | } | ||
71 | |||
72 | if (dssdev->ops->hdmi.set_hdmi_mode) | ||
73 | dssdev->ops->hdmi.set_hdmi_mode(dssdev, hdmi_mode); | ||
74 | |||
75 | if (hdmi_mode && dssdev->ops->hdmi.set_infoframe) { | ||
76 | struct hdmi_avi_infoframe avi; | ||
77 | int r; | ||
78 | |||
79 | r = drm_hdmi_avi_infoframe_from_display_mode(&avi, adjusted_mode, | ||
80 | false); | ||
81 | if (r == 0) | ||
82 | dssdev->ops->hdmi.set_infoframe(dssdev, &avi); | ||
83 | } | ||
84 | } | ||
85 | |||
55 | static void omap_encoder_mode_set(struct drm_encoder *encoder, | 86 | static void omap_encoder_mode_set(struct drm_encoder *encoder, |
56 | struct drm_display_mode *mode, | 87 | struct drm_display_mode *mode, |
57 | struct drm_display_mode *adjusted_mode) | 88 | struct drm_display_mode *adjusted_mode) |
58 | { | 89 | { |
59 | struct drm_device *dev = encoder->dev; | ||
60 | struct omap_encoder *omap_encoder = to_omap_encoder(encoder); | 90 | struct omap_encoder *omap_encoder = to_omap_encoder(encoder); |
61 | struct drm_connector *connector; | ||
62 | struct omap_dss_device *dssdev; | 91 | struct omap_dss_device *dssdev; |
63 | struct videomode vm = { 0 }; | 92 | struct videomode vm = { 0 }; |
64 | bool hdmi_mode; | ||
65 | int r; | ||
66 | 93 | ||
67 | drm_display_mode_to_videomode(adjusted_mode, &vm); | 94 | drm_display_mode_to_videomode(adjusted_mode, &vm); |
68 | 95 | ||
@@ -112,27 +139,8 @@ static void omap_encoder_mode_set(struct drm_encoder *encoder, | |||
112 | } | 139 | } |
113 | 140 | ||
114 | /* Set the HDMI mode and HDMI infoframe if applicable. */ | 141 | /* Set the HDMI mode and HDMI infoframe if applicable. */ |
115 | hdmi_mode = false; | 142 | if (omap_encoder->output->output_type == OMAP_DISPLAY_TYPE_HDMI) |
116 | list_for_each_entry(connector, &dev->mode_config.connector_list, head) { | 143 | omap_encoder_hdmi_mode_set(encoder, adjusted_mode); |
117 | if (connector->encoder == encoder) { | ||
118 | hdmi_mode = omap_connector_get_hdmi_mode(connector); | ||
119 | break; | ||
120 | } | ||
121 | } | ||
122 | |||
123 | dssdev = omap_encoder->output; | ||
124 | |||
125 | if (dssdev->ops->hdmi.set_hdmi_mode) | ||
126 | dssdev->ops->hdmi.set_hdmi_mode(dssdev, hdmi_mode); | ||
127 | |||
128 | if (hdmi_mode && dssdev->ops->hdmi.set_infoframe) { | ||
129 | struct hdmi_avi_infoframe avi; | ||
130 | |||
131 | r = drm_hdmi_avi_infoframe_from_display_mode(&avi, adjusted_mode, | ||
132 | false); | ||
133 | if (r == 0) | ||
134 | dssdev->ops->hdmi.set_infoframe(dssdev, &avi); | ||
135 | } | ||
136 | } | 144 | } |
137 | 145 | ||
138 | static void omap_encoder_disable(struct drm_encoder *encoder) | 146 | static void omap_encoder_disable(struct drm_encoder *encoder) |