@@ -586,6 +586,25 @@ drm_atomic_helper_check_modeset(struct drm_device *dev,
}
}
+ for_each_oldnew_connector_in_state(state, connector, old_connector_state, new_connector_state, i) {
+ const struct drm_connector_helper_funcs *conn_funcs;
+ struct drm_crtc_state *crtc_state;
+
+ conn_funcs = connector->helper_private;
+ if (!conn_funcs->atomic_release)
+ continue;
+
+ if (!old_connector_state->crtc)
+ continue;
+
+ crtc_state = drm_atomic_get_existing_crtc_state(state, old_connector_state->crtc);
+
+ if (crtc_state->connectors_changed ||
+ crtc_state->mode_changed ||
+ (crtc_state->active_changed && !crtc_state->active))
+ conn_funcs->atomic_release(connector, new_connector_state);
+ }
+
return mode_fixup(state);
}
EXPORT_SYMBOL(drm_atomic_helper_check_modeset);
@@ -836,6 +836,22 @@ struct drm_connector_helper_funcs {
*/
struct drm_encoder *(*atomic_best_encoder)(struct drm_connector *connector,
struct drm_connector_state *connector_state);
+
+ /**
+ * @atomic_release:
+ *
+ * This function is conditionally called to release shared resources
+ * when the attached CRTC's mode changes or it's connectors change or
+ * becomes inactive. It is called before the corresponding
+ * &drm_crtc_helper_funcs.atomic_check or
+ * &drm_crtc_helper_funcs.mode_fixup hooks are called.
+ *
+ * NOTE:
+ *
+ * This function is called in the check phase of an atomic update.
+ */
+ void (*atomic_release)(struct drm_connector *connector,
+ struct drm_connector_state *connector_state);
};
/**