From: Ankit Nautiyal <ankit.k.nautiyal@xxxxxxxxx> If the user mode does not support aspect-ratio, and requests for a modeset, then the flag bits representing aspect ratio in the given user-mode must be rejected. Similarly, while preparing a user-mode from kernel mode, the aspect-ratio info must not be given, if aspect-ratio is not supported by the user. This patch: 1. adds a new bit field aspect_ratio_allowed in drm_atomic_state, which is set only if the aspect-ratio is supported by the user. 2. discards the aspect-ratio info from the user mode while preparing kernel mode structure, during modeset, if the user does not support aspect ratio. 3. avoids setting the aspect-ratio info in user-mode, while converting user-mode from the kernel-mode. Signed-off-by: Ankit Nautiyal <ankit.k.nautiyal@xxxxxxxxx> V3: Addressed review comments from Ville: -Do not corrupt the current crtc state by updating aspect ratio on the fly. --- drivers/gpu/drm/drm_atomic.c | 61 +++++++++++++++++++++++++++++++++++++++++--- drivers/gpu/drm/drm_crtc.c | 19 ++++++++++++++ include/drm/drm_atomic.h | 2 ++ 3 files changed, 79 insertions(+), 3 deletions(-) diff --git a/drivers/gpu/drm/drm_atomic.c b/drivers/gpu/drm/drm_atomic.c index 69ff763..39313e2 100644 --- a/drivers/gpu/drm/drm_atomic.c +++ b/drivers/gpu/drm/drm_atomic.c @@ -316,6 +316,35 @@ static s32 __user *get_out_fence_for_crtc(struct drm_atomic_state *state, return fence_ptr; } +/** + * drm_atomic_allow_aspect_ratio_for_kmode + * @state: the crtc state + * @mode: kernel-internal mode, which is used to create a duplicate mode, + * with updated picture aspect ratio. + * + * Allow the aspect ratio info in the kernel mode only if aspect ratio is + * supported. + * + * RETURNS: + * kernel-internal mode with updated picture aspect ratio value. + */ + +struct drm_display_mode* +drm_atomic_allow_aspect_ratio_for_kmode(struct drm_crtc_state *state, + const struct drm_display_mode *mode) +{ + struct drm_atomic_state *atomic_state = state->state; + struct drm_display_mode *ar_kmode; + + ar_kmode = drm_mode_duplicate(state->crtc->dev, mode); + /* + * If aspect ratio is not supported, set the picture aspect ratio as + * NONE. + */ + if (atomic_state && !atomic_state->allow_aspect_ratio) + ar_kmode->picture_aspect_ratio = HDMI_PICTURE_ASPECT_NONE; + return ar_kmode; +} /** * drm_atomic_set_mode_for_crtc - set mode for CRTC @@ -341,7 +370,10 @@ int drm_atomic_set_mode_for_crtc(struct drm_crtc_state *state, state->mode_blob = NULL; if (mode) { - drm_mode_convert_to_umode(&umode, mode); + struct drm_display_mode *ar_mode; + + ar_mode = drm_atomic_allow_aspect_ratio_for_kmode(state, mode); + drm_mode_convert_to_umode(&umode, ar_mode); state->mode_blob = drm_property_create_blob(state->crtc->dev, sizeof(umode), @@ -349,10 +381,11 @@ int drm_atomic_set_mode_for_crtc(struct drm_crtc_state *state, if (IS_ERR(state->mode_blob)) return PTR_ERR(state->mode_blob); - drm_mode_copy(&state->mode, mode); + drm_mode_copy(&state->mode, ar_mode); state->enable = true; DRM_DEBUG_ATOMIC("Set [MODE:%s] for CRTC state %p\n", - mode->name, state); + ar_mode->name, state); + drm_mode_destroy(state->crtc->dev, ar_mode); } else { memset(&state->mode, 0, sizeof(state->mode)); state->enable = false; @@ -436,6 +469,25 @@ drm_atomic_replace_property_blob_from_id(struct drm_device *dev, } /** + * drm_atomic_allow_aspect_ratio_for_umode + * @state: the crtc state + * @umode: userspace mode, whose aspect ratio flag bits are to be updated. + * + * Allow the aspect ratio info in the userspace mode only if aspect ratio is + * supported. + */ +void +drm_atomic_allow_aspect_ratio_for_umode(struct drm_crtc_state *state, + struct drm_mode_modeinfo *umode) +{ + struct drm_atomic_state *atomic_state = state->state; + + /* Reset the aspect ratio flags if aspect ratio is not supported */ + if (atomic_state && !atomic_state->allow_aspect_ratio) + umode->flags &= (~DRM_MODE_FLAG_PIC_AR_MASK); +} + +/** * drm_atomic_crtc_set_property - set property on CRTC * @crtc: the drm CRTC to set a property on * @state: the state object to update with the new property value @@ -464,6 +516,9 @@ int drm_atomic_crtc_set_property(struct drm_crtc *crtc, else if (property == config->prop_mode_id) { struct drm_property_blob *mode = drm_property_lookup_blob(dev, val); + drm_atomic_allow_aspect_ratio_for_umode(state, + (struct drm_mode_modeinfo *) + mode->data); ret = drm_atomic_set_mode_prop_for_crtc(state, mode); drm_property_blob_put(mode); return ret; diff --git a/drivers/gpu/drm/drm_crtc.c b/drivers/gpu/drm/drm_crtc.c index f0556e6..a2d34fa 100644 --- a/drivers/gpu/drm/drm_crtc.c +++ b/drivers/gpu/drm/drm_crtc.c @@ -425,6 +425,13 @@ int drm_mode_getcrtc(struct drm_device *dev, drm_modeset_lock(&crtc->mutex, NULL); if (crtc->state) { if (crtc->state->enable) { + /* + * If the aspect-ratio is not required by the, + * userspace, do not set the aspect-ratio flags. + */ + if (!file_priv->aspect_ratio_allowed) + crtc->state->mode.picture_aspect_ratio = + HDMI_PICTURE_ASPECT_NONE; drm_mode_convert_to_umode(&crtc_resp->mode, &crtc->state->mode); crtc_resp->mode_valid = 1; @@ -435,6 +442,13 @@ int drm_mode_getcrtc(struct drm_device *dev, crtc_resp->x = crtc->x; crtc_resp->y = crtc->y; if (crtc->enabled) { + /* + * If the aspect-ratio is not required by the, + * userspace, do not set the aspect-ratio flags. + */ + if (!file_priv->aspect_ratio_allowed) + crtc->mode.picture_aspect_ratio = + HDMI_PICTURE_ASPECT_NONE; drm_mode_convert_to_umode(&crtc_resp->mode, &crtc->mode); crtc_resp->mode_valid = 1; @@ -610,6 +624,11 @@ int drm_mode_setcrtc(struct drm_device *dev, void *data, goto out; } + /* If the user does not ask for aspect ratio, reset the aspect + * ratio bits in the usermode. + */ + if (!file_priv->aspect_ratio_allowed) + crtc_req->mode.flags &= (~DRM_MODE_FLAG_PIC_AR_MASK); ret = drm_mode_convert_umode(mode, &crtc_req->mode); if (ret) { DRM_DEBUG_KMS("Invalid mode\n"); diff --git a/include/drm/drm_atomic.h b/include/drm/drm_atomic.h index 1c27526..130dad9 100644 --- a/include/drm/drm_atomic.h +++ b/include/drm/drm_atomic.h @@ -237,6 +237,7 @@ struct __drm_private_objs_state { * @ref: count of all references to this state (will not be freed until zero) * @dev: parent DRM device * @allow_modeset: allow full modeset + * @allow_aspect_ratio: allow the aspect ratio info to be passes to user * @legacy_cursor_update: hint to enforce legacy cursor IOCTL semantics * @async_update: hint for asynchronous plane update * @planes: pointer to array of structures with per-plane data @@ -256,6 +257,7 @@ struct drm_atomic_state { struct drm_device *dev; bool allow_modeset : 1; + bool allow_aspect_ratio : 1; bool legacy_cursor_update : 1; bool async_update : 1; struct __drm_planes_state *planes; -- 2.7.4 _______________________________________________ dri-devel mailing list dri-devel@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/dri-devel