2014-07-25 23:30:38 +04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Red Hat
|
|
|
|
* Copyright (C) 2014 Intel Corp.
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
|
|
* to deal in the Software without restriction, including without limitation
|
|
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
|
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
|
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*
|
|
|
|
* Authors:
|
|
|
|
* Rob Clark <robdclark@gmail.com>
|
|
|
|
* Daniel Vetter <daniel.vetter@ffwll.ch>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef DRM_ATOMIC_H_
|
|
|
|
#define DRM_ATOMIC_H_
|
|
|
|
|
2014-11-25 14:09:48 +03:00
|
|
|
#include <drm/drm_crtc.h>
|
|
|
|
|
2014-07-25 23:30:38 +04:00
|
|
|
struct drm_atomic_state * __must_check
|
|
|
|
drm_atomic_state_alloc(struct drm_device *dev);
|
|
|
|
void drm_atomic_state_clear(struct drm_atomic_state *state);
|
|
|
|
void drm_atomic_state_free(struct drm_atomic_state *state);
|
2015-05-18 11:06:40 +03:00
|
|
|
|
|
|
|
int __must_check
|
|
|
|
drm_atomic_state_init(struct drm_device *dev, struct drm_atomic_state *state);
|
|
|
|
void drm_atomic_state_default_clear(struct drm_atomic_state *state);
|
|
|
|
void drm_atomic_state_default_release(struct drm_atomic_state *state);
|
2014-07-25 23:30:38 +04:00
|
|
|
|
|
|
|
struct drm_crtc_state * __must_check
|
|
|
|
drm_atomic_get_crtc_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_crtc *crtc);
|
2014-12-19 00:01:46 +03:00
|
|
|
int drm_atomic_crtc_set_property(struct drm_crtc *crtc,
|
|
|
|
struct drm_crtc_state *state, struct drm_property *property,
|
|
|
|
uint64_t val);
|
2014-07-25 23:30:38 +04:00
|
|
|
struct drm_plane_state * __must_check
|
|
|
|
drm_atomic_get_plane_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_plane *plane);
|
2014-12-19 00:01:46 +03:00
|
|
|
int drm_atomic_plane_set_property(struct drm_plane *plane,
|
|
|
|
struct drm_plane_state *state, struct drm_property *property,
|
|
|
|
uint64_t val);
|
2014-07-25 23:30:38 +04:00
|
|
|
struct drm_connector_state * __must_check
|
|
|
|
drm_atomic_get_connector_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_connector *connector);
|
2014-12-19 00:01:46 +03:00
|
|
|
int drm_atomic_connector_set_property(struct drm_connector *connector,
|
|
|
|
struct drm_connector_state *state, struct drm_property *property,
|
|
|
|
uint64_t val);
|
2014-12-19 00:01:50 +03:00
|
|
|
|
2015-05-13 11:37:25 +03:00
|
|
|
/**
|
|
|
|
* drm_atomic_get_existing_crtc_state - get crtc state, if it exists
|
|
|
|
* @state: global atomic state object
|
|
|
|
* @crtc: crtc to grab
|
|
|
|
*
|
|
|
|
* This function returns the crtc state for the given crtc, or NULL
|
|
|
|
* if the crtc is not part of the global atomic state.
|
|
|
|
*/
|
|
|
|
static inline struct drm_crtc_state *
|
|
|
|
drm_atomic_get_existing_crtc_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_crtc *crtc)
|
|
|
|
{
|
|
|
|
return state->crtc_states[drm_crtc_index(crtc)];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_atomic_get_existing_plane_state - get plane state, if it exists
|
|
|
|
* @state: global atomic state object
|
|
|
|
* @plane: plane to grab
|
|
|
|
*
|
|
|
|
* This function returns the plane state for the given plane, or NULL
|
|
|
|
* if the plane is not part of the global atomic state.
|
|
|
|
*/
|
|
|
|
static inline struct drm_plane_state *
|
|
|
|
drm_atomic_get_existing_plane_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_plane *plane)
|
|
|
|
{
|
2016-06-02 01:06:33 +03:00
|
|
|
return state->planes[drm_plane_index(plane)].state;
|
2015-05-13 11:37:25 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* drm_atomic_get_existing_connector_state - get connector state, if it exists
|
|
|
|
* @state: global atomic state object
|
|
|
|
* @connector: connector to grab
|
|
|
|
*
|
|
|
|
* This function returns the connector state for the given connector,
|
|
|
|
* or NULL if the connector is not part of the global atomic state.
|
|
|
|
*/
|
|
|
|
static inline struct drm_connector_state *
|
|
|
|
drm_atomic_get_existing_connector_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_connector *connector)
|
|
|
|
{
|
|
|
|
int index = drm_connector_index(connector);
|
|
|
|
|
|
|
|
if (index >= state->num_connector)
|
|
|
|
return NULL;
|
|
|
|
|
2016-06-02 01:06:32 +03:00
|
|
|
return state->connectors[index].state;
|
2015-05-13 11:37:25 +03:00
|
|
|
}
|
|
|
|
|
2016-06-02 17:21:44 +03:00
|
|
|
/**
|
|
|
|
* __drm_atomic_get_current_plane_state - get current plane state
|
|
|
|
* @state: global atomic state object
|
|
|
|
* @plane: plane to grab
|
|
|
|
*
|
|
|
|
* This function returns the plane state for the given plane, either from
|
|
|
|
* @state, or if the plane isn't part of the atomic state update, from @plane.
|
|
|
|
* This is useful in atomic check callbacks, when drivers need to peek at, but
|
|
|
|
* not change, state of other planes, since it avoids threading an error code
|
|
|
|
* back up the call chain.
|
|
|
|
*
|
|
|
|
* WARNING:
|
|
|
|
*
|
|
|
|
* Note that this function is in general unsafe since it doesn't check for the
|
|
|
|
* required locking for access state structures. Drivers must ensure that it is
|
|
|
|
* save to access the returned state structure through other means. One commone
|
|
|
|
* example is when planes are fixed to a single CRTC, and the driver knows that
|
|
|
|
* the CRTC locks is held already. In that case holding the CRTC locks gives a
|
|
|
|
* read-lock on all planes connected to that CRTC. But if planes can be
|
|
|
|
* reassigned things get more tricky. In that case it's better to use
|
|
|
|
* drm_atomic_get_plane_state and wire up full error handling.
|
|
|
|
*
|
|
|
|
* Returns:
|
|
|
|
*
|
|
|
|
* Read-only pointer to the current plane state.
|
|
|
|
*/
|
|
|
|
static inline const struct drm_plane_state *
|
|
|
|
__drm_atomic_get_current_plane_state(struct drm_atomic_state *state,
|
|
|
|
struct drm_plane *plane)
|
|
|
|
{
|
2016-06-02 01:06:33 +03:00
|
|
|
if (state->planes[drm_plane_index(plane)].state)
|
|
|
|
return state->planes[drm_plane_index(plane)].state;
|
2016-06-02 17:21:44 +03:00
|
|
|
|
|
|
|
return plane->state;
|
|
|
|
}
|
|
|
|
|
2015-05-26 16:36:48 +03:00
|
|
|
int __must_check
|
|
|
|
drm_atomic_set_mode_for_crtc(struct drm_crtc_state *state,
|
|
|
|
struct drm_display_mode *mode);
|
2014-07-25 23:30:38 +04:00
|
|
|
int __must_check
|
2015-05-25 21:11:52 +03:00
|
|
|
drm_atomic_set_mode_prop_for_crtc(struct drm_crtc_state *state,
|
|
|
|
struct drm_property_blob *blob);
|
|
|
|
int __must_check
|
2014-11-27 17:49:39 +03:00
|
|
|
drm_atomic_set_crtc_for_plane(struct drm_plane_state *plane_state,
|
|
|
|
struct drm_crtc *crtc);
|
2014-11-05 00:57:27 +03:00
|
|
|
void drm_atomic_set_fb_for_plane(struct drm_plane_state *plane_state,
|
|
|
|
struct drm_framebuffer *fb);
|
2014-07-25 23:30:38 +04:00
|
|
|
int __must_check
|
|
|
|
drm_atomic_set_crtc_for_connector(struct drm_connector_state *conn_state,
|
|
|
|
struct drm_crtc *crtc);
|
|
|
|
int __must_check
|
|
|
|
drm_atomic_add_affected_connectors(struct drm_atomic_state *state,
|
|
|
|
struct drm_crtc *crtc);
|
2015-05-19 17:41:02 +03:00
|
|
|
int __must_check
|
|
|
|
drm_atomic_add_affected_planes(struct drm_atomic_state *state,
|
|
|
|
struct drm_crtc *crtc);
|
|
|
|
|
2014-07-25 23:30:38 +04:00
|
|
|
void drm_atomic_legacy_backoff(struct drm_atomic_state *state);
|
|
|
|
|
2015-11-11 13:29:09 +03:00
|
|
|
void
|
|
|
|
drm_atomic_clean_old_fb(struct drm_device *dev, unsigned plane_mask, int ret);
|
|
|
|
|
2014-07-25 23:30:38 +04:00
|
|
|
int __must_check drm_atomic_check_only(struct drm_atomic_state *state);
|
|
|
|
int __must_check drm_atomic_commit(struct drm_atomic_state *state);
|
2016-04-26 17:11:35 +03:00
|
|
|
int __must_check drm_atomic_nonblocking_commit(struct drm_atomic_state *state);
|
2014-07-25 23:30:38 +04:00
|
|
|
|
2016-06-02 01:06:32 +03:00
|
|
|
#define for_each_connector_in_state(__state, connector, connector_state, __i) \
|
2015-04-10 14:58:39 +03:00
|
|
|
for ((__i) = 0; \
|
2016-06-02 01:06:32 +03:00
|
|
|
(__i) < (__state)->num_connector && \
|
|
|
|
((connector) = (__state)->connectors[__i].ptr, \
|
|
|
|
(connector_state) = (__state)->connectors[__i].state, 1); \
|
2015-04-10 14:58:39 +03:00
|
|
|
(__i)++) \
|
drm: fix potential dangling else problems in for_each_ macros
We have serious dangling else bugs waiting to happen in our for_each_
style macros with ifs. Consider, for example,
#define drm_for_each_plane_mask(plane, dev, plane_mask) \
list_for_each_entry((plane), &(dev)->mode_config.plane_list, head) \
if ((plane_mask) & (1 << drm_plane_index(plane)))
If this is used in context:
if (condition)
drm_for_each_plane_mask(plane, dev, plane_mask);
else
foo();
foo() will be called for each plane *not* in plane_mask, if condition
holds, and not at all if condition doesn't hold.
Fix this by reversing the conditions in the macros, and adding an else
branch for the "for each" block, so that other if/else blocks can't
interfere. Provide a "for_each_if" helper macro to make it easier to get
this right.
Signed-off-by: Jani Nikula <jani.nikula@intel.com>
Link: http://patchwork.freedesktop.org/patch/msgid/1448392916-2281-1-git-send-email-jani.nikula@intel.com
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
2015-11-24 22:21:55 +03:00
|
|
|
for_each_if (connector)
|
2015-04-10 14:58:39 +03:00
|
|
|
|
|
|
|
#define for_each_crtc_in_state(state, crtc, crtc_state, __i) \
|
|
|
|
for ((__i) = 0; \
|
2015-05-25 13:29:44 +03:00
|
|
|
(__i) < (state)->dev->mode_config.num_crtc && \
|
|
|
|
((crtc) = (state)->crtcs[__i], \
|
|
|
|
(crtc_state) = (state)->crtc_states[__i], 1); \
|
2015-04-10 14:58:39 +03:00
|
|
|
(__i)++) \
|
drm: fix potential dangling else problems in for_each_ macros
We have serious dangling else bugs waiting to happen in our for_each_
style macros with ifs. Consider, for example,
#define drm_for_each_plane_mask(plane, dev, plane_mask) \
list_for_each_entry((plane), &(dev)->mode_config.plane_list, head) \
if ((plane_mask) & (1 << drm_plane_index(plane)))
If this is used in context:
if (condition)
drm_for_each_plane_mask(plane, dev, plane_mask);
else
foo();
foo() will be called for each plane *not* in plane_mask, if condition
holds, and not at all if condition doesn't hold.
Fix this by reversing the conditions in the macros, and adding an else
branch for the "for each" block, so that other if/else blocks can't
interfere. Provide a "for_each_if" helper macro to make it easier to get
this right.
Signed-off-by: Jani Nikula <jani.nikula@intel.com>
Link: http://patchwork.freedesktop.org/patch/msgid/1448392916-2281-1-git-send-email-jani.nikula@intel.com
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
2015-11-24 22:21:55 +03:00
|
|
|
for_each_if (crtc_state)
|
2015-04-10 14:58:39 +03:00
|
|
|
|
2016-06-02 01:06:33 +03:00
|
|
|
#define for_each_plane_in_state(__state, plane, plane_state, __i) \
|
2015-05-25 13:29:44 +03:00
|
|
|
for ((__i) = 0; \
|
2016-06-02 01:06:33 +03:00
|
|
|
(__i) < (__state)->dev->mode_config.num_total_plane && \
|
|
|
|
((plane) = (__state)->planes[__i].ptr, \
|
|
|
|
(plane_state) = (__state)->planes[__i].state, 1); \
|
2015-05-25 13:29:44 +03:00
|
|
|
(__i)++) \
|
drm: fix potential dangling else problems in for_each_ macros
We have serious dangling else bugs waiting to happen in our for_each_
style macros with ifs. Consider, for example,
#define drm_for_each_plane_mask(plane, dev, plane_mask) \
list_for_each_entry((plane), &(dev)->mode_config.plane_list, head) \
if ((plane_mask) & (1 << drm_plane_index(plane)))
If this is used in context:
if (condition)
drm_for_each_plane_mask(plane, dev, plane_mask);
else
foo();
foo() will be called for each plane *not* in plane_mask, if condition
holds, and not at all if condition doesn't hold.
Fix this by reversing the conditions in the macros, and adding an else
branch for the "for each" block, so that other if/else blocks can't
interfere. Provide a "for_each_if" helper macro to make it easier to get
this right.
Signed-off-by: Jani Nikula <jani.nikula@intel.com>
Link: http://patchwork.freedesktop.org/patch/msgid/1448392916-2281-1-git-send-email-jani.nikula@intel.com
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
2015-11-24 22:21:55 +03:00
|
|
|
for_each_if (plane_state)
|
2015-06-18 10:58:55 +03:00
|
|
|
static inline bool
|
|
|
|
drm_atomic_crtc_needs_modeset(struct drm_crtc_state *state)
|
|
|
|
{
|
2015-07-21 14:28:57 +03:00
|
|
|
return state->mode_changed || state->active_changed ||
|
|
|
|
state->connectors_changed;
|
2015-06-18 10:58:55 +03:00
|
|
|
}
|
|
|
|
|
2015-04-10 14:58:39 +03:00
|
|
|
|
2014-07-25 23:30:38 +04:00
|
|
|
#endif /* DRM_ATOMIC_H_ */
|