212 lines
5.5 KiB
C
212 lines
5.5 KiB
C
|
// SPDX-License-Identifier: GPL-2.0
|
||
|
/*
|
||
|
* Copyright (C) 2018 Texas Instruments Incorporated - http://www.ti.com/
|
||
|
* Author: Benoit Parrot <bparrot@ti.com>
|
||
|
*/
|
||
|
|
||
|
#include <drm/drm_atomic.h>
|
||
|
#include <drm/drm_atomic_helper.h>
|
||
|
|
||
|
#include "omap_dmm_tiler.h"
|
||
|
#include "omap_drv.h"
|
||
|
|
||
|
/*
|
||
|
* overlay funcs
|
||
|
*/
|
||
|
static const char * const overlay_id_to_name[] = {
|
||
|
[OMAP_DSS_GFX] = "gfx",
|
||
|
[OMAP_DSS_VIDEO1] = "vid1",
|
||
|
[OMAP_DSS_VIDEO2] = "vid2",
|
||
|
[OMAP_DSS_VIDEO3] = "vid3",
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* Find a free overlay with the required caps and supported fourcc
|
||
|
*/
|
||
|
static struct omap_hw_overlay *
|
||
|
omap_plane_find_free_overlay(struct drm_device *dev, struct drm_plane *hwoverlay_to_plane[],
|
||
|
u32 caps, u32 fourcc)
|
||
|
{
|
||
|
struct omap_drm_private *priv = dev->dev_private;
|
||
|
int i;
|
||
|
|
||
|
DBG("caps: %x fourcc: %x", caps, fourcc);
|
||
|
|
||
|
for (i = 0; i < priv->num_ovls; i++) {
|
||
|
struct omap_hw_overlay *cur = priv->overlays[i];
|
||
|
|
||
|
DBG("%d: id: %d cur->caps: %x",
|
||
|
cur->idx, cur->id, cur->caps);
|
||
|
|
||
|
/* skip if already in-use */
|
||
|
if (hwoverlay_to_plane[cur->idx])
|
||
|
continue;
|
||
|
|
||
|
/* skip if doesn't support some required caps: */
|
||
|
if (caps & ~cur->caps)
|
||
|
continue;
|
||
|
|
||
|
/* check supported format */
|
||
|
if (!dispc_ovl_color_mode_supported(priv->dispc,
|
||
|
cur->id, fourcc))
|
||
|
continue;
|
||
|
|
||
|
return cur;
|
||
|
}
|
||
|
|
||
|
DBG("no match");
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Assign a new overlay to a plane with the required caps and supported fourcc
|
||
|
* If a plane need a new overlay, the previous one should have been released
|
||
|
* with omap_overlay_release()
|
||
|
* This should be called from the plane atomic_check() in order to prepare the
|
||
|
* next global overlay_map to be enabled when atomic transaction is valid.
|
||
|
*/
|
||
|
int omap_overlay_assign(struct drm_atomic_state *s, struct drm_plane *plane,
|
||
|
u32 caps, u32 fourcc, struct omap_hw_overlay **overlay,
|
||
|
struct omap_hw_overlay **r_overlay)
|
||
|
{
|
||
|
/* Get the global state of the current atomic transaction */
|
||
|
struct omap_global_state *state = omap_get_global_state(s);
|
||
|
struct drm_plane **overlay_map = state->hwoverlay_to_plane;
|
||
|
struct omap_hw_overlay *ovl, *r_ovl;
|
||
|
|
||
|
ovl = omap_plane_find_free_overlay(s->dev, overlay_map, caps, fourcc);
|
||
|
if (!ovl)
|
||
|
return -ENOMEM;
|
||
|
|
||
|
overlay_map[ovl->idx] = plane;
|
||
|
*overlay = ovl;
|
||
|
|
||
|
if (r_overlay) {
|
||
|
r_ovl = omap_plane_find_free_overlay(s->dev, overlay_map,
|
||
|
caps, fourcc);
|
||
|
if (!r_ovl) {
|
||
|
overlay_map[ovl->idx] = NULL;
|
||
|
*overlay = NULL;
|
||
|
return -ENOMEM;
|
||
|
}
|
||
|
|
||
|
overlay_map[r_ovl->idx] = plane;
|
||
|
*r_overlay = r_ovl;
|
||
|
}
|
||
|
|
||
|
DBG("%s: assign to plane %s caps %x", ovl->name, plane->name, caps);
|
||
|
|
||
|
if (r_overlay) {
|
||
|
DBG("%s: assign to right of plane %s caps %x",
|
||
|
r_ovl->name, plane->name, caps);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Release an overlay from a plane if the plane gets not visible or the plane
|
||
|
* need a new overlay if overlay caps changes.
|
||
|
* This should be called from the plane atomic_check() in order to prepare the
|
||
|
* next global overlay_map to be enabled when atomic transaction is valid.
|
||
|
*/
|
||
|
void omap_overlay_release(struct drm_atomic_state *s, struct omap_hw_overlay *overlay)
|
||
|
{
|
||
|
/* Get the global state of the current atomic transaction */
|
||
|
struct omap_global_state *state = omap_get_global_state(s);
|
||
|
struct drm_plane **overlay_map = state->hwoverlay_to_plane;
|
||
|
|
||
|
if (!overlay)
|
||
|
return;
|
||
|
|
||
|
if (WARN_ON(!overlay_map[overlay->idx]))
|
||
|
return;
|
||
|
|
||
|
DBG("%s: release from plane %s", overlay->name, overlay_map[overlay->idx]->name);
|
||
|
|
||
|
overlay_map[overlay->idx] = NULL;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Update an overlay state that was attached to a plane before the current atomic state.
|
||
|
* This should be called from the plane atomic_update() or atomic_disable(),
|
||
|
* where an overlay association to a plane could have changed between the old and current
|
||
|
* atomic state.
|
||
|
*/
|
||
|
void omap_overlay_update_state(struct omap_drm_private *priv,
|
||
|
struct omap_hw_overlay *overlay)
|
||
|
{
|
||
|
struct omap_global_state *state = omap_get_existing_global_state(priv);
|
||
|
struct drm_plane **overlay_map = state->hwoverlay_to_plane;
|
||
|
|
||
|
/* Check if this overlay is not used anymore, then disable it */
|
||
|
if (!overlay_map[overlay->idx]) {
|
||
|
DBG("%s: disabled", overlay->name);
|
||
|
|
||
|
/* disable the overlay */
|
||
|
dispc_ovl_enable(priv->dispc, overlay->id, false);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void omap_overlay_destroy(struct omap_hw_overlay *overlay)
|
||
|
{
|
||
|
kfree(overlay);
|
||
|
}
|
||
|
|
||
|
static struct omap_hw_overlay *omap_overlay_init(enum omap_plane_id overlay_id,
|
||
|
enum omap_overlay_caps caps)
|
||
|
{
|
||
|
struct omap_hw_overlay *overlay;
|
||
|
|
||
|
overlay = kzalloc(sizeof(*overlay), GFP_KERNEL);
|
||
|
if (!overlay)
|
||
|
return ERR_PTR(-ENOMEM);
|
||
|
|
||
|
overlay->name = overlay_id_to_name[overlay_id];
|
||
|
overlay->id = overlay_id;
|
||
|
overlay->caps = caps;
|
||
|
|
||
|
return overlay;
|
||
|
}
|
||
|
|
||
|
int omap_hwoverlays_init(struct omap_drm_private *priv)
|
||
|
{
|
||
|
static const enum omap_plane_id hw_plane_ids[] = {
|
||
|
OMAP_DSS_GFX, OMAP_DSS_VIDEO1,
|
||
|
OMAP_DSS_VIDEO2, OMAP_DSS_VIDEO3,
|
||
|
};
|
||
|
u32 num_overlays = dispc_get_num_ovls(priv->dispc);
|
||
|
enum omap_overlay_caps caps;
|
||
|
int i, ret;
|
||
|
|
||
|
for (i = 0; i < num_overlays; i++) {
|
||
|
struct omap_hw_overlay *overlay;
|
||
|
|
||
|
caps = dispc_ovl_get_caps(priv->dispc, hw_plane_ids[i]);
|
||
|
overlay = omap_overlay_init(hw_plane_ids[i], caps);
|
||
|
if (IS_ERR(overlay)) {
|
||
|
ret = PTR_ERR(overlay);
|
||
|
dev_err(priv->dev, "failed to construct overlay for %s (%d)\n",
|
||
|
overlay_id_to_name[i], ret);
|
||
|
omap_hwoverlays_destroy(priv);
|
||
|
return ret;
|
||
|
}
|
||
|
overlay->idx = priv->num_ovls;
|
||
|
priv->overlays[priv->num_ovls++] = overlay;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
void omap_hwoverlays_destroy(struct omap_drm_private *priv)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
for (i = 0; i < priv->num_ovls; i++) {
|
||
|
omap_overlay_destroy(priv->overlays[i]);
|
||
|
priv->overlays[i] = NULL;
|
||
|
}
|
||
|
|
||
|
priv->num_ovls = 0;
|
||
|
}
|