Commit de3916c7 authored by Drew Davenport's avatar Drew Davenport Committed by Rob Clark

drm/msm/dpu: Track resources in global state

Move mapping of resources to encoder ids from the resource manager to a
new dpu_global_state struct. Store this struct in global atomic state.

Before this patch, atomic test would be performed by modifying global
state (resource manager), and backing out any changes if the test fails.
By using drm atomic global state, this is not necessary as any changes
to the global state will be discarded if the test fails.
Signed-off-by: default avatarDrew Davenport <ddavenport@chromium.org>
Signed-off-by: default avatarRob Clark <robdclark@chromium.org>
parent bb00a452
...@@ -164,7 +164,6 @@ enum dpu_enc_rc_states { ...@@ -164,7 +164,6 @@ enum dpu_enc_rc_states {
* clks and resources after IDLE_TIMEOUT time. * clks and resources after IDLE_TIMEOUT time.
* @vsync_event_work: worker to handle vsync event for autorefresh * @vsync_event_work: worker to handle vsync event for autorefresh
* @topology: topology of the display * @topology: topology of the display
* @mode_set_complete: flag to indicate modeset completion
* @idle_timeout: idle timeout duration in milliseconds * @idle_timeout: idle timeout duration in milliseconds
*/ */
struct dpu_encoder_virt { struct dpu_encoder_virt {
...@@ -202,7 +201,6 @@ struct dpu_encoder_virt { ...@@ -202,7 +201,6 @@ struct dpu_encoder_virt {
struct delayed_work delayed_off_work; struct delayed_work delayed_off_work;
struct kthread_work vsync_event_work; struct kthread_work vsync_event_work;
struct msm_display_topology topology; struct msm_display_topology topology;
bool mode_set_complete;
u32 idle_timeout; u32 idle_timeout;
}; };
...@@ -563,6 +561,7 @@ static int dpu_encoder_virt_atomic_check( ...@@ -563,6 +561,7 @@ static int dpu_encoder_virt_atomic_check(
const struct drm_display_mode *mode; const struct drm_display_mode *mode;
struct drm_display_mode *adj_mode; struct drm_display_mode *adj_mode;
struct msm_display_topology topology; struct msm_display_topology topology;
struct dpu_global_state *global_state;
int i = 0; int i = 0;
int ret = 0; int ret = 0;
...@@ -579,6 +578,7 @@ static int dpu_encoder_virt_atomic_check( ...@@ -579,6 +578,7 @@ static int dpu_encoder_virt_atomic_check(
dpu_kms = to_dpu_kms(priv->kms); dpu_kms = to_dpu_kms(priv->kms);
mode = &crtc_state->mode; mode = &crtc_state->mode;
adj_mode = &crtc_state->adjusted_mode; adj_mode = &crtc_state->adjusted_mode;
global_state = dpu_kms_get_existing_global_state(dpu_kms);
trace_dpu_enc_atomic_check(DRMID(drm_enc)); trace_dpu_enc_atomic_check(DRMID(drm_enc));
/* /*
...@@ -610,17 +610,15 @@ static int dpu_encoder_virt_atomic_check( ...@@ -610,17 +610,15 @@ static int dpu_encoder_virt_atomic_check(
topology = dpu_encoder_get_topology(dpu_enc, dpu_kms, adj_mode); topology = dpu_encoder_get_topology(dpu_enc, dpu_kms, adj_mode);
/* Reserve dynamic resources now. Indicating AtomicTest phase */ /* Reserve dynamic resources now. */
if (!ret) { if (!ret) {
/* /*
* Avoid reserving resources when mode set is pending. Topology * Avoid reserving resources when mode set is pending. Topology
* info may not be available to complete reservation. * info may not be available to complete reservation.
*/ */
if (drm_atomic_crtc_needs_modeset(crtc_state) if (drm_atomic_crtc_needs_modeset(crtc_state)) {
&& dpu_enc->mode_set_complete) { ret = dpu_rm_reserve(&dpu_kms->rm, global_state,
ret = dpu_rm_reserve(&dpu_kms->rm, drm_enc, crtc_state, drm_enc, crtc_state, topology);
topology, true);
dpu_enc->mode_set_complete = false;
} }
} }
...@@ -957,12 +955,13 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc, ...@@ -957,12 +955,13 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc,
struct drm_connector *conn = NULL, *conn_iter; struct drm_connector *conn = NULL, *conn_iter;
struct drm_crtc *drm_crtc; struct drm_crtc *drm_crtc;
struct dpu_crtc_state *cstate; struct dpu_crtc_state *cstate;
struct dpu_global_state *global_state;
struct msm_display_topology topology; struct msm_display_topology topology;
struct dpu_hw_blk *hw_pp[MAX_CHANNELS_PER_ENC]; struct dpu_hw_blk *hw_pp[MAX_CHANNELS_PER_ENC];
struct dpu_hw_blk *hw_ctl[MAX_CHANNELS_PER_ENC]; struct dpu_hw_blk *hw_ctl[MAX_CHANNELS_PER_ENC];
struct dpu_hw_blk *hw_lm[MAX_CHANNELS_PER_ENC]; struct dpu_hw_blk *hw_lm[MAX_CHANNELS_PER_ENC];
int num_lm, num_ctl, num_pp; int num_lm, num_ctl, num_pp;
int i, j, ret; int i, j;
if (!drm_enc) { if (!drm_enc) {
DPU_ERROR("invalid encoder\n"); DPU_ERROR("invalid encoder\n");
...@@ -976,6 +975,12 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc, ...@@ -976,6 +975,12 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc,
dpu_kms = to_dpu_kms(priv->kms); dpu_kms = to_dpu_kms(priv->kms);
connector_list = &dpu_kms->dev->mode_config.connector_list; connector_list = &dpu_kms->dev->mode_config.connector_list;
global_state = dpu_kms_get_existing_global_state(dpu_kms);
if (IS_ERR_OR_NULL(global_state)) {
DPU_ERROR("Failed to get global state");
return;
}
trace_dpu_enc_mode_set(DRMID(drm_enc)); trace_dpu_enc_mode_set(DRMID(drm_enc));
list_for_each_entry(conn_iter, connector_list, head) list_for_each_entry(conn_iter, connector_list, head)
...@@ -996,21 +1001,14 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc, ...@@ -996,21 +1001,14 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc,
topology = dpu_encoder_get_topology(dpu_enc, dpu_kms, adj_mode); topology = dpu_encoder_get_topology(dpu_enc, dpu_kms, adj_mode);
/* Reserve dynamic resources now. Indicating non-AtomicTest phase */ /* Query resource that have been reserved in atomic check step. */
ret = dpu_rm_reserve(&dpu_kms->rm, drm_enc, drm_crtc->state, num_pp = dpu_rm_get_assigned_resources(&dpu_kms->rm, global_state,
topology, false); drm_enc->base.id, DPU_HW_BLK_PINGPONG, hw_pp,
if (ret) { ARRAY_SIZE(hw_pp));
DPU_ERROR_ENC(dpu_enc, num_ctl = dpu_rm_get_assigned_resources(&dpu_kms->rm, global_state,
"failed to reserve hw resources, %d\n", ret); drm_enc->base.id, DPU_HW_BLK_CTL, hw_ctl, ARRAY_SIZE(hw_ctl));
return; num_lm = dpu_rm_get_assigned_resources(&dpu_kms->rm, global_state,
} drm_enc->base.id, DPU_HW_BLK_LM, hw_lm, ARRAY_SIZE(hw_lm));
num_pp = dpu_rm_get_assigned_resources(&dpu_kms->rm, drm_enc->base.id,
DPU_HW_BLK_PINGPONG, hw_pp, ARRAY_SIZE(hw_pp));
num_ctl = dpu_rm_get_assigned_resources(&dpu_kms->rm, drm_enc->base.id,
DPU_HW_BLK_CTL, hw_ctl, ARRAY_SIZE(hw_ctl));
num_lm = dpu_rm_get_assigned_resources(&dpu_kms->rm, drm_enc->base.id,
DPU_HW_BLK_LM, hw_lm, ARRAY_SIZE(hw_lm));
for (i = 0; i < MAX_CHANNELS_PER_ENC; i++) for (i = 0; i < MAX_CHANNELS_PER_ENC; i++)
dpu_enc->hw_pp[i] = i < num_pp ? to_dpu_hw_pingpong(hw_pp[i]) dpu_enc->hw_pp[i] = i < num_pp ? to_dpu_hw_pingpong(hw_pp[i])
...@@ -1035,21 +1033,21 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc, ...@@ -1035,21 +1033,21 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc,
if (!dpu_enc->hw_pp[i]) { if (!dpu_enc->hw_pp[i]) {
DPU_ERROR_ENC(dpu_enc, DPU_ERROR_ENC(dpu_enc,
"no pp block assigned at idx: %d\n", i); "no pp block assigned at idx: %d\n", i);
goto error; return;
} }
if (!hw_ctl[i]) { if (!hw_ctl[i]) {
DPU_ERROR_ENC(dpu_enc, DPU_ERROR_ENC(dpu_enc,
"no ctl block assigned at idx: %d\n", i); "no ctl block assigned at idx: %d\n", i);
goto error; return;
} }
phys->hw_pp = dpu_enc->hw_pp[i]; phys->hw_pp = dpu_enc->hw_pp[i];
phys->hw_ctl = to_dpu_hw_ctl(hw_ctl[i]); phys->hw_ctl = to_dpu_hw_ctl(hw_ctl[i]);
num_blk = dpu_rm_get_assigned_resources(&dpu_kms->rm, num_blk = dpu_rm_get_assigned_resources(&dpu_kms->rm,
drm_enc->base.id, DPU_HW_BLK_INTF, hw_blk, global_state, drm_enc->base.id, DPU_HW_BLK_INTF,
ARRAY_SIZE(hw_blk)); hw_blk, ARRAY_SIZE(hw_blk));
for (j = 0; j < num_blk; j++) { for (j = 0; j < num_blk; j++) {
struct dpu_hw_intf *hw_intf; struct dpu_hw_intf *hw_intf;
...@@ -1061,18 +1059,13 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc, ...@@ -1061,18 +1059,13 @@ static void dpu_encoder_virt_mode_set(struct drm_encoder *drm_enc,
if (!phys->hw_intf) { if (!phys->hw_intf) {
DPU_ERROR_ENC(dpu_enc, DPU_ERROR_ENC(dpu_enc,
"no intf block assigned at idx: %d\n", i); "no intf block assigned at idx: %d\n", i);
goto error; return;
} }
phys->connector = conn->state->connector; phys->connector = conn->state->connector;
if (phys->ops.mode_set) if (phys->ops.mode_set)
phys->ops.mode_set(phys, mode, adj_mode); phys->ops.mode_set(phys, mode, adj_mode);
} }
dpu_enc->mode_set_complete = true;
error:
dpu_rm_release(&dpu_kms->rm, drm_enc);
} }
static void _dpu_encoder_virt_enable_helper(struct drm_encoder *drm_enc) static void _dpu_encoder_virt_enable_helper(struct drm_encoder *drm_enc)
...@@ -1169,6 +1162,7 @@ static void dpu_encoder_virt_disable(struct drm_encoder *drm_enc) ...@@ -1169,6 +1162,7 @@ static void dpu_encoder_virt_disable(struct drm_encoder *drm_enc)
struct dpu_encoder_virt *dpu_enc = NULL; struct dpu_encoder_virt *dpu_enc = NULL;
struct msm_drm_private *priv; struct msm_drm_private *priv;
struct dpu_kms *dpu_kms; struct dpu_kms *dpu_kms;
struct dpu_global_state *global_state;
int i = 0; int i = 0;
if (!drm_enc) { if (!drm_enc) {
...@@ -1187,6 +1181,7 @@ static void dpu_encoder_virt_disable(struct drm_encoder *drm_enc) ...@@ -1187,6 +1181,7 @@ static void dpu_encoder_virt_disable(struct drm_encoder *drm_enc)
priv = drm_enc->dev->dev_private; priv = drm_enc->dev->dev_private;
dpu_kms = to_dpu_kms(priv->kms); dpu_kms = to_dpu_kms(priv->kms);
global_state = dpu_kms_get_existing_global_state(dpu_kms);
trace_dpu_enc_disable(DRMID(drm_enc)); trace_dpu_enc_disable(DRMID(drm_enc));
...@@ -1216,7 +1211,7 @@ static void dpu_encoder_virt_disable(struct drm_encoder *drm_enc) ...@@ -1216,7 +1211,7 @@ static void dpu_encoder_virt_disable(struct drm_encoder *drm_enc)
DPU_DEBUG_ENC(dpu_enc, "encoder disabled\n"); DPU_DEBUG_ENC(dpu_enc, "encoder disabled\n");
dpu_rm_release(&dpu_kms->rm, drm_enc); dpu_rm_release(global_state, drm_enc);
mutex_unlock(&dpu_enc->enc_lock); mutex_unlock(&dpu_enc->enc_lock);
} }
......
...@@ -228,6 +228,85 @@ static int dpu_kms_debugfs_init(struct msm_kms *kms, struct drm_minor *minor) ...@@ -228,6 +228,85 @@ static int dpu_kms_debugfs_init(struct msm_kms *kms, struct drm_minor *minor)
} }
#endif #endif
/* Global/shared object state funcs */
/*
* This is a helper that returns the private state currently in operation.
* Note that this would return the "old_state" if called in the atomic check
* path, and the "new_state" after the atomic swap has been done.
*/
struct dpu_global_state *
dpu_kms_get_existing_global_state(struct dpu_kms *dpu_kms)
{
return to_dpu_global_state(dpu_kms->global_state.state);
}
/*
* This acquires the modeset lock set aside for global state, creates
* a new duplicated private object state.
*/
struct dpu_global_state *dpu_kms_get_global_state(struct drm_atomic_state *s)
{
struct msm_drm_private *priv = s->dev->dev_private;
struct dpu_kms *dpu_kms = to_dpu_kms(priv->kms);
struct drm_private_state *priv_state;
int ret;
ret = drm_modeset_lock(&dpu_kms->global_state_lock, s->acquire_ctx);
if (ret)
return ERR_PTR(ret);
priv_state = drm_atomic_get_private_obj_state(s,
&dpu_kms->global_state);
if (IS_ERR(priv_state))
return ERR_CAST(priv_state);
return to_dpu_global_state(priv_state);
}
static struct drm_private_state *
dpu_kms_global_duplicate_state(struct drm_private_obj *obj)
{
struct dpu_global_state *state;
state = kmemdup(obj->state, sizeof(*state), GFP_KERNEL);
if (!state)
return NULL;
__drm_atomic_helper_private_obj_duplicate_state(obj, &state->base);
return &state->base;
}
static void dpu_kms_global_destroy_state(struct drm_private_obj *obj,
struct drm_private_state *state)
{
struct dpu_global_state *dpu_state = to_dpu_global_state(state);
kfree(dpu_state);
}
static const struct drm_private_state_funcs dpu_kms_global_state_funcs = {
.atomic_duplicate_state = dpu_kms_global_duplicate_state,
.atomic_destroy_state = dpu_kms_global_destroy_state,
};
static int dpu_kms_global_obj_init(struct dpu_kms *dpu_kms)
{
struct dpu_global_state *state;
drm_modeset_lock_init(&dpu_kms->global_state_lock);
state = kzalloc(sizeof(*state), GFP_KERNEL);
if (!state)
return -ENOMEM;
drm_atomic_private_obj_init(dpu_kms->dev, &dpu_kms->global_state,
&state->base,
&dpu_kms_global_state_funcs);
return 0;
}
static int dpu_kms_enable_vblank(struct msm_kms *kms, struct drm_crtc *crtc) static int dpu_kms_enable_vblank(struct msm_kms *kms, struct drm_crtc *crtc)
{ {
return dpu_crtc_vblank(crtc, true); return dpu_crtc_vblank(crtc, true);
...@@ -770,6 +849,11 @@ static int dpu_kms_hw_init(struct msm_kms *kms) ...@@ -770,6 +849,11 @@ static int dpu_kms_hw_init(struct msm_kms *kms)
dpu_kms = to_dpu_kms(kms); dpu_kms = to_dpu_kms(kms);
dev = dpu_kms->dev; dev = dpu_kms->dev;
rc = dpu_kms_global_obj_init(dpu_kms);
if (rc)
return rc;
priv = dev->dev_private; priv = dev->dev_private;
atomic_set(&dpu_kms->bandwidth_ref, 0); atomic_set(&dpu_kms->bandwidth_ref, 0);
......
...@@ -111,6 +111,13 @@ struct dpu_kms { ...@@ -111,6 +111,13 @@ struct dpu_kms {
struct dpu_core_perf perf; struct dpu_core_perf perf;
/*
* Global private object state, Do not access directly, use
* dpu_kms_global_get_state()
*/
struct drm_modeset_lock global_state_lock;
struct drm_private_obj global_state;
struct dpu_rm rm; struct dpu_rm rm;
bool rm_init; bool rm_init;
...@@ -139,6 +146,25 @@ struct vsync_info { ...@@ -139,6 +146,25 @@ struct vsync_info {
#define to_dpu_kms(x) container_of(x, struct dpu_kms, base) #define to_dpu_kms(x) container_of(x, struct dpu_kms, base)
#define to_dpu_global_state(x) container_of(x, struct dpu_global_state, base)
/* Global private object state for tracking resources that are shared across
* multiple kms objects (planes/crtcs/etc).
*/
struct dpu_global_state {
struct drm_private_state base;
uint32_t pingpong_to_enc_id[PINGPONG_MAX - PINGPONG_0];
uint32_t mixer_to_enc_id[LM_MAX - LM_0];
uint32_t ctl_to_enc_id[CTL_MAX - CTL_0];
uint32_t intf_to_enc_id[INTF_MAX - INTF_0];
};
struct dpu_global_state
*dpu_kms_get_existing_global_state(struct dpu_kms *dpu_kms);
struct dpu_global_state
*__must_check dpu_kms_get_global_state(struct drm_atomic_state *s);
/** /**
* Debugfs functions - extra helper functions for debugfs support * Debugfs functions - extra helper functions for debugfs support
* *
......
...@@ -66,8 +66,6 @@ int dpu_rm_destroy(struct dpu_rm *rm) ...@@ -66,8 +66,6 @@ int dpu_rm_destroy(struct dpu_rm *rm)
} }
} }
mutex_destroy(&rm->rm_lock);
return 0; return 0;
} }
...@@ -85,8 +83,6 @@ int dpu_rm_init(struct dpu_rm *rm, ...@@ -85,8 +83,6 @@ int dpu_rm_init(struct dpu_rm *rm,
/* Clear, setup lists */ /* Clear, setup lists */
memset(rm, 0, sizeof(*rm)); memset(rm, 0, sizeof(*rm));
mutex_init(&rm->rm_lock);
/* Interrogate HW catalog and create tracking items for hw blocks */ /* Interrogate HW catalog and create tracking items for hw blocks */
for (i = 0; i < cat->mixer_count; i++) { for (i = 0; i < cat->mixer_count; i++) {
struct dpu_hw_mixer *hw; struct dpu_hw_mixer *hw;
...@@ -230,13 +226,14 @@ static bool _dpu_rm_check_lm_peer(struct dpu_rm *rm, int primary_idx, ...@@ -230,13 +226,14 @@ static bool _dpu_rm_check_lm_peer(struct dpu_rm *rm, int primary_idx,
* @Return: true if lm matches all requirements, false otherwise * @Return: true if lm matches all requirements, false otherwise
*/ */
static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm, static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm,
struct dpu_global_state *global_state,
uint32_t enc_id, int lm_idx, int *pp_idx) uint32_t enc_id, int lm_idx, int *pp_idx)
{ {
const struct dpu_lm_cfg *lm_cfg; const struct dpu_lm_cfg *lm_cfg;
int idx; int idx;
/* Already reserved? */ /* Already reserved? */
if (reserved_by_other(rm->mixer_to_enc_id, lm_idx, enc_id)) { if (reserved_by_other(global_state->mixer_to_enc_id, lm_idx, enc_id)) {
DPU_DEBUG("lm %d already reserved\n", lm_idx + LM_0); DPU_DEBUG("lm %d already reserved\n", lm_idx + LM_0);
return false; return false;
} }
...@@ -248,7 +245,7 @@ static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm, ...@@ -248,7 +245,7 @@ static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm,
return false; return false;
} }
if (reserved_by_other(rm->pingpong_to_enc_id, idx, enc_id)) { if (reserved_by_other(global_state->pingpong_to_enc_id, idx, enc_id)) {
DPU_DEBUG("lm %d pp %d already reserved\n", lm_cfg->id, DPU_DEBUG("lm %d pp %d already reserved\n", lm_cfg->id,
lm_cfg->pingpong); lm_cfg->pingpong);
return false; return false;
...@@ -257,7 +254,9 @@ static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm, ...@@ -257,7 +254,9 @@ static bool _dpu_rm_check_lm_and_get_connected_blks(struct dpu_rm *rm,
return true; return true;
} }
static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id, static int _dpu_rm_reserve_lms(struct dpu_rm *rm,
struct dpu_global_state *global_state,
uint32_t enc_id,
struct dpu_rm_requirements *reqs) struct dpu_rm_requirements *reqs)
{ {
...@@ -279,8 +278,8 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id, ...@@ -279,8 +278,8 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id,
lm_count = 0; lm_count = 0;
lm_idx[lm_count] = i; lm_idx[lm_count] = i;
if (!_dpu_rm_check_lm_and_get_connected_blks( if (!_dpu_rm_check_lm_and_get_connected_blks(rm, global_state,
rm, enc_id, i, &pp_idx[lm_count])) { enc_id, i, &pp_idx[lm_count])) {
continue; continue;
} }
...@@ -298,8 +297,9 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id, ...@@ -298,8 +297,9 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id,
continue; continue;
} }
if (!_dpu_rm_check_lm_and_get_connected_blks( if (!_dpu_rm_check_lm_and_get_connected_blks(rm,
rm, enc_id, j, &pp_idx[lm_count])) { global_state, enc_id, j,
&pp_idx[lm_count])) {
continue; continue;
} }
...@@ -314,8 +314,8 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id, ...@@ -314,8 +314,8 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id,
} }
for (i = 0; i < lm_count; i++) { for (i = 0; i < lm_count; i++) {
rm->mixer_to_enc_id[lm_idx[i]] = enc_id; global_state->mixer_to_enc_id[lm_idx[i]] = enc_id;
rm->pingpong_to_enc_id[pp_idx[i]] = enc_id; global_state->pingpong_to_enc_id[pp_idx[i]] = enc_id;
trace_dpu_rm_reserve_lms(lm_idx[i] + LM_0, enc_id, trace_dpu_rm_reserve_lms(lm_idx[i] + LM_0, enc_id,
pp_idx[i] + PINGPONG_0); pp_idx[i] + PINGPONG_0);
...@@ -326,6 +326,7 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id, ...@@ -326,6 +326,7 @@ static int _dpu_rm_reserve_lms(struct dpu_rm *rm, uint32_t enc_id,
static int _dpu_rm_reserve_ctls( static int _dpu_rm_reserve_ctls(
struct dpu_rm *rm, struct dpu_rm *rm,
struct dpu_global_state *global_state,
uint32_t enc_id, uint32_t enc_id,
const struct msm_display_topology *top) const struct msm_display_topology *top)
{ {
...@@ -345,7 +346,7 @@ static int _dpu_rm_reserve_ctls( ...@@ -345,7 +346,7 @@ static int _dpu_rm_reserve_ctls(
if (!rm->ctl_blks[j]) if (!rm->ctl_blks[j])
continue; continue;
if (reserved_by_other(rm->ctl_to_enc_id, j, enc_id)) if (reserved_by_other(global_state->ctl_to_enc_id, j, enc_id))
continue; continue;
ctl = to_dpu_hw_ctl(rm->ctl_blks[j]); ctl = to_dpu_hw_ctl(rm->ctl_blks[j]);
...@@ -369,7 +370,7 @@ static int _dpu_rm_reserve_ctls( ...@@ -369,7 +370,7 @@ static int _dpu_rm_reserve_ctls(
return -ENAVAIL; return -ENAVAIL;
for (i = 0; i < ARRAY_SIZE(ctl_idx) && i < num_ctls; i++) { for (i = 0; i < ARRAY_SIZE(ctl_idx) && i < num_ctls; i++) {
rm->ctl_to_enc_id[ctl_idx[i]] = enc_id; global_state->ctl_to_enc_id[ctl_idx[i]] = enc_id;
trace_dpu_rm_reserve_ctls(i + CTL_0, enc_id); trace_dpu_rm_reserve_ctls(i + CTL_0, enc_id);
} }
...@@ -378,27 +379,34 @@ static int _dpu_rm_reserve_ctls( ...@@ -378,27 +379,34 @@ static int _dpu_rm_reserve_ctls(
static int _dpu_rm_reserve_intf( static int _dpu_rm_reserve_intf(
struct dpu_rm *rm, struct dpu_rm *rm,
struct dpu_global_state *global_state,
uint32_t enc_id, uint32_t enc_id,
uint32_t id) uint32_t id)
{ {
int idx = id - INTF_0; int idx = id - INTF_0;
if (idx < 0 || idx >= ARRAY_SIZE(rm->intf_blks)) {
DPU_ERROR("invalid intf id: %d", id);
return -EINVAL;
}
if (!rm->intf_blks[idx]) { if (!rm->intf_blks[idx]) {
DPU_ERROR("couldn't find intf id %d\n", id); DPU_ERROR("couldn't find intf id %d\n", id);
return -EINVAL; return -EINVAL;
} }
if (reserved_by_other(rm->intf_to_enc_id, idx, enc_id)) { if (reserved_by_other(global_state->intf_to_enc_id, idx, enc_id)) {
DPU_ERROR("intf id %d already reserved\n", id); DPU_ERROR("intf id %d already reserved\n", id);
return -ENAVAIL; return -ENAVAIL;
} }
rm->intf_to_enc_id[idx] = enc_id; global_state->intf_to_enc_id[idx] = enc_id;
return 0; return 0;
} }
static int _dpu_rm_reserve_intf_related_hw( static int _dpu_rm_reserve_intf_related_hw(
struct dpu_rm *rm, struct dpu_rm *rm,
struct dpu_global_state *global_state,
uint32_t enc_id, uint32_t enc_id,
struct dpu_encoder_hw_resources *hw_res) struct dpu_encoder_hw_resources *hw_res)
{ {
...@@ -409,7 +417,7 @@ static int _dpu_rm_reserve_intf_related_hw( ...@@ -409,7 +417,7 @@ static int _dpu_rm_reserve_intf_related_hw(
if (hw_res->intfs[i] == INTF_MODE_NONE) if (hw_res->intfs[i] == INTF_MODE_NONE)
continue; continue;
id = i + INTF_0; id = i + INTF_0;
ret = _dpu_rm_reserve_intf(rm, enc_id, id); ret = _dpu_rm_reserve_intf(rm, global_state, enc_id, id);
if (ret) if (ret)
return ret; return ret;
} }
...@@ -419,24 +427,27 @@ static int _dpu_rm_reserve_intf_related_hw( ...@@ -419,24 +427,27 @@ static int _dpu_rm_reserve_intf_related_hw(
static int _dpu_rm_make_reservation( static int _dpu_rm_make_reservation(
struct dpu_rm *rm, struct dpu_rm *rm,
struct dpu_global_state *global_state,
struct drm_encoder *enc, struct drm_encoder *enc,
struct dpu_rm_requirements *reqs) struct dpu_rm_requirements *reqs)
{ {
int ret; int ret;
ret = _dpu_rm_reserve_lms(rm, enc->base.id, reqs); ret = _dpu_rm_reserve_lms(rm, global_state, enc->base.id, reqs);
if (ret) { if (ret) {
DPU_ERROR("unable to find appropriate mixers\n"); DPU_ERROR("unable to find appropriate mixers\n");
return ret; return ret;
} }
ret = _dpu_rm_reserve_ctls(rm, enc->base.id, &reqs->topology); ret = _dpu_rm_reserve_ctls(rm, global_state, enc->base.id,
&reqs->topology);
if (ret) { if (ret) {
DPU_ERROR("unable to find appropriate CTL\n"); DPU_ERROR("unable to find appropriate CTL\n");
return ret; return ret;
} }
ret = _dpu_rm_reserve_intf_related_hw(rm, enc->base.id, &reqs->hw_res); ret = _dpu_rm_reserve_intf_related_hw(rm, global_state, enc->base.id,
&reqs->hw_res);
if (ret) if (ret)
return ret; return ret;
...@@ -470,33 +481,25 @@ static void _dpu_rm_clear_mapping(uint32_t *res_mapping, int cnt, ...@@ -470,33 +481,25 @@ static void _dpu_rm_clear_mapping(uint32_t *res_mapping, int cnt,
} }
} }
static void _dpu_rm_release_reservation(struct dpu_rm *rm, uint32_t enc_id) void dpu_rm_release(struct dpu_global_state *global_state,
{ struct drm_encoder *enc)
_dpu_rm_clear_mapping(rm->pingpong_to_enc_id,
ARRAY_SIZE(rm->pingpong_to_enc_id), enc_id);
_dpu_rm_clear_mapping(rm->mixer_to_enc_id,
ARRAY_SIZE(rm->mixer_to_enc_id), enc_id);
_dpu_rm_clear_mapping(rm->ctl_to_enc_id,
ARRAY_SIZE(rm->ctl_to_enc_id), enc_id);
_dpu_rm_clear_mapping(rm->intf_to_enc_id,
ARRAY_SIZE(rm->intf_to_enc_id), enc_id);
}
void dpu_rm_release(struct dpu_rm *rm, struct drm_encoder *enc)
{ {
mutex_lock(&rm->rm_lock); _dpu_rm_clear_mapping(global_state->pingpong_to_enc_id,
ARRAY_SIZE(global_state->pingpong_to_enc_id), enc->base.id);
_dpu_rm_release_reservation(rm, enc->base.id); _dpu_rm_clear_mapping(global_state->mixer_to_enc_id,
ARRAY_SIZE(global_state->mixer_to_enc_id), enc->base.id);
mutex_unlock(&rm->rm_lock); _dpu_rm_clear_mapping(global_state->ctl_to_enc_id,
ARRAY_SIZE(global_state->ctl_to_enc_id), enc->base.id);
_dpu_rm_clear_mapping(global_state->intf_to_enc_id,
ARRAY_SIZE(global_state->intf_to_enc_id), enc->base.id);
} }
int dpu_rm_reserve( int dpu_rm_reserve(
struct dpu_rm *rm, struct dpu_rm *rm,
struct dpu_global_state *global_state,
struct drm_encoder *enc, struct drm_encoder *enc,
struct drm_crtc_state *crtc_state, struct drm_crtc_state *crtc_state,
struct msm_display_topology topology, struct msm_display_topology topology)
bool test_only)
{ {
struct dpu_rm_requirements reqs; struct dpu_rm_requirements reqs;
int ret; int ret;
...@@ -505,35 +508,31 @@ int dpu_rm_reserve( ...@@ -505,35 +508,31 @@ int dpu_rm_reserve(
if (!drm_atomic_crtc_needs_modeset(crtc_state)) if (!drm_atomic_crtc_needs_modeset(crtc_state))
return 0; return 0;
DRM_DEBUG_KMS("reserving hw for enc %d crtc %d test_only %d\n", if (IS_ERR(global_state)) {
enc->base.id, crtc_state->crtc->base.id, test_only); DPU_ERROR("failed to global state\n");
return PTR_ERR(global_state);
}
mutex_lock(&rm->rm_lock); DRM_DEBUG_KMS("reserving hw for enc %d crtc %d\n",
enc->base.id, crtc_state->crtc->base.id);
ret = _dpu_rm_populate_requirements(enc, &reqs, topology); ret = _dpu_rm_populate_requirements(enc, &reqs, topology);
if (ret) { if (ret) {
DPU_ERROR("failed to populate hw requirements\n"); DPU_ERROR("failed to populate hw requirements\n");
goto end; return ret;
} }
ret = _dpu_rm_make_reservation(rm, enc, &reqs); ret = _dpu_rm_make_reservation(rm, global_state, enc, &reqs);
if (ret) { if (ret)
DPU_ERROR("failed to reserve hw resources: %d\n", ret); DPU_ERROR("failed to reserve hw resources: %d\n", ret);
_dpu_rm_release_reservation(rm, enc->base.id);
} else if (test_only) {
/* test_only: test the reservation and then undo */
DPU_DEBUG("test_only: discard test [enc: %d]\n",
enc->base.id);
_dpu_rm_release_reservation(rm, enc->base.id);
}
end:
mutex_unlock(&rm->rm_lock);
return ret; return ret;
} }
int dpu_rm_get_assigned_resources(struct dpu_rm *rm, uint32_t enc_id, int dpu_rm_get_assigned_resources(struct dpu_rm *rm,
struct dpu_global_state *global_state, uint32_t enc_id,
enum dpu_hw_blk_type type, struct dpu_hw_blk **blks, int blks_size) enum dpu_hw_blk_type type, struct dpu_hw_blk **blks, int blks_size)
{ {
struct dpu_hw_blk **hw_blks; struct dpu_hw_blk **hw_blks;
...@@ -543,22 +542,22 @@ int dpu_rm_get_assigned_resources(struct dpu_rm *rm, uint32_t enc_id, ...@@ -543,22 +542,22 @@ int dpu_rm_get_assigned_resources(struct dpu_rm *rm, uint32_t enc_id,
switch (type) { switch (type) {
case DPU_HW_BLK_PINGPONG: case DPU_HW_BLK_PINGPONG:
hw_blks = rm->pingpong_blks; hw_blks = rm->pingpong_blks;
hw_to_enc_id = rm->pingpong_to_enc_id; hw_to_enc_id = global_state->pingpong_to_enc_id;
max_blks = ARRAY_SIZE(rm->pingpong_blks); max_blks = ARRAY_SIZE(rm->pingpong_blks);
break; break;
case DPU_HW_BLK_LM: case DPU_HW_BLK_LM:
hw_blks = rm->mixer_blks; hw_blks = rm->mixer_blks;
hw_to_enc_id = rm->mixer_to_enc_id; hw_to_enc_id = global_state->mixer_to_enc_id;
max_blks = ARRAY_SIZE(rm->mixer_blks); max_blks = ARRAY_SIZE(rm->mixer_blks);
break; break;
case DPU_HW_BLK_CTL: case DPU_HW_BLK_CTL:
hw_blks = rm->ctl_blks; hw_blks = rm->ctl_blks;
hw_to_enc_id = rm->ctl_to_enc_id; hw_to_enc_id = global_state->ctl_to_enc_id;
max_blks = ARRAY_SIZE(rm->ctl_blks); max_blks = ARRAY_SIZE(rm->ctl_blks);
break; break;
case DPU_HW_BLK_INTF: case DPU_HW_BLK_INTF:
hw_blks = rm->intf_blks; hw_blks = rm->intf_blks;
hw_to_enc_id = rm->intf_to_enc_id; hw_to_enc_id = global_state->intf_to_enc_id;
max_blks = ARRAY_SIZE(rm->intf_blks); max_blks = ARRAY_SIZE(rm->intf_blks);
break; break;
default: default:
......
...@@ -11,6 +11,7 @@ ...@@ -11,6 +11,7 @@
#include "msm_kms.h" #include "msm_kms.h"
#include "dpu_hw_top.h" #include "dpu_hw_top.h"
struct dpu_global_state;
/** /**
* struct dpu_rm - DPU dynamic hardware resource manager * struct dpu_rm - DPU dynamic hardware resource manager
...@@ -18,10 +19,6 @@ ...@@ -18,10 +19,6 @@
* @mixer_blks: array of layer mixer hardware resources * @mixer_blks: array of layer mixer hardware resources
* @ctl_blks: array of ctl hardware resources * @ctl_blks: array of ctl hardware resources
* @intf_blks: array of intf hardware resources * @intf_blks: array of intf hardware resources
* @pingpong_to_enc_id: mapping of pingpong hardware resources to an encoder ID
* @mixer_to_enc_id: mapping of mixer hardware resources to an encoder ID
* @ctl_to_enc_id: mapping of ctl hardware resources to an encoder ID
* @intf_to_enc_id: mapping of intf hardware resources to an encoder ID
* @lm_max_width: cached layer mixer maximum width * @lm_max_width: cached layer mixer maximum width
* @rm_lock: resource manager mutex * @rm_lock: resource manager mutex
*/ */
...@@ -31,13 +28,7 @@ struct dpu_rm { ...@@ -31,13 +28,7 @@ struct dpu_rm {
struct dpu_hw_blk *ctl_blks[CTL_MAX - CTL_0]; struct dpu_hw_blk *ctl_blks[CTL_MAX - CTL_0];
struct dpu_hw_blk *intf_blks[INTF_MAX - INTF_0]; struct dpu_hw_blk *intf_blks[INTF_MAX - INTF_0];
uint32_t pingpong_to_enc_id[PINGPONG_MAX - PINGPONG_0];
uint32_t mixer_to_enc_id[LM_MAX - LM_0];
uint32_t ctl_to_enc_id[CTL_MAX - CTL_0];
uint32_t intf_to_enc_id[INTF_MAX - INTF_0];
uint32_t lm_max_width; uint32_t lm_max_width;
struct mutex rm_lock;
}; };
/** /**
...@@ -70,14 +61,13 @@ int dpu_rm_destroy(struct dpu_rm *rm); ...@@ -70,14 +61,13 @@ int dpu_rm_destroy(struct dpu_rm *rm);
* @drm_enc: DRM Encoder handle * @drm_enc: DRM Encoder handle
* @crtc_state: Proposed Atomic DRM CRTC State handle * @crtc_state: Proposed Atomic DRM CRTC State handle
* @topology: Pointer to topology info for the display * @topology: Pointer to topology info for the display
* @test_only: Atomic-Test phase, discard results (unless property overrides)
* @Return: 0 on Success otherwise -ERROR * @Return: 0 on Success otherwise -ERROR
*/ */
int dpu_rm_reserve(struct dpu_rm *rm, int dpu_rm_reserve(struct dpu_rm *rm,
struct dpu_global_state *global_state,
struct drm_encoder *drm_enc, struct drm_encoder *drm_enc,
struct drm_crtc_state *crtc_state, struct drm_crtc_state *crtc_state,
struct msm_display_topology topology, struct msm_display_topology topology);
bool test_only);
/** /**
* dpu_rm_reserve - Given the encoder for the display chain, release any * dpu_rm_reserve - Given the encoder for the display chain, release any
...@@ -86,12 +76,14 @@ int dpu_rm_reserve(struct dpu_rm *rm, ...@@ -86,12 +76,14 @@ int dpu_rm_reserve(struct dpu_rm *rm,
* @enc: DRM Encoder handle * @enc: DRM Encoder handle
* @Return: 0 on Success otherwise -ERROR * @Return: 0 on Success otherwise -ERROR
*/ */
void dpu_rm_release(struct dpu_rm *rm, struct drm_encoder *enc); void dpu_rm_release(struct dpu_global_state *global_state,
struct drm_encoder *enc);
/** /**
* Get hw resources of the given type that are assigned to this encoder. * Get hw resources of the given type that are assigned to this encoder.
*/ */
int dpu_rm_get_assigned_resources(struct dpu_rm *rm, uint32_t enc_id, int dpu_rm_get_assigned_resources(struct dpu_rm *rm,
struct dpu_global_state *global_state, uint32_t enc_id,
enum dpu_hw_blk_type type, struct dpu_hw_blk **blks, int blks_size); enum dpu_hw_blk_type type, struct dpu_hw_blk **blks, int blks_size);
#endif /* __DPU_RM_H__ */ #endif /* __DPU_RM_H__ */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment