Commit 45324a26 authored by Archit Taneja's avatar Archit Taneja Committed by Tomi Valkeinen

OMAPDSS: APPLY: Add manager timings as extra_info in private data

DISPC manager size and DISPC manager blanking parameters(for LCD managers)
follow the shadow register programming model. Currently, they are programmed
directly by the interface drivers.

To configure manager timings using APPLY, there is a need to introduce extra
info flags for managers, similar to what is done for overlays. This is needed
because timings aren't a part of overlay_manager_info struct configured by a
user of DSS, they are configured internally by the interface or panel drivers.

Add dirty and shadow_dirty extra_info flags for managers, update these flags
at the appropriate places. Rewrite the function extra_info_update_ongoing()
slightly as checking for manager's extra_info flags can simplify the code a bit.

Create function dss_mgr_set_timings() which applies the new manager timings to
extra_info.
Signed-off-by: default avatarArchit Taneja <archit@ti.com>
parent 408d9dbb
...@@ -99,6 +99,11 @@ struct mgr_priv_data { ...@@ -99,6 +99,11 @@ struct mgr_priv_data {
/* If true, a display is enabled using this manager */ /* If true, a display is enabled using this manager */
bool enabled; bool enabled;
bool extra_info_dirty;
bool shadow_extra_info_dirty;
struct omap_video_timings timings;
}; };
static struct { static struct {
...@@ -261,6 +266,20 @@ static bool need_isr(void) ...@@ -261,6 +266,20 @@ static bool need_isr(void)
if (mp->shadow_info_dirty) if (mp->shadow_info_dirty)
return true; return true;
/*
* NOTE: we don't check extra_info flags for disabled
* managers, once the manager is enabled, the extra_info
* related manager changes will be taken in by HW.
*/
/* to write new values to registers */
if (mp->extra_info_dirty)
return true;
/* to set GO bit */
if (mp->shadow_extra_info_dirty)
return true;
list_for_each_entry(ovl, &mgr->overlays, list) { list_for_each_entry(ovl, &mgr->overlays, list) {
struct ovl_priv_data *op; struct ovl_priv_data *op;
...@@ -305,7 +324,7 @@ static bool need_go(struct omap_overlay_manager *mgr) ...@@ -305,7 +324,7 @@ static bool need_go(struct omap_overlay_manager *mgr)
mp = get_mgr_priv(mgr); mp = get_mgr_priv(mgr);
if (mp->shadow_info_dirty) if (mp->shadow_info_dirty || mp->shadow_extra_info_dirty)
return true; return true;
list_for_each_entry(ovl, &mgr->overlays, list) { list_for_each_entry(ovl, &mgr->overlays, list) {
...@@ -320,20 +339,16 @@ static bool need_go(struct omap_overlay_manager *mgr) ...@@ -320,20 +339,16 @@ static bool need_go(struct omap_overlay_manager *mgr)
/* returns true if an extra_info field is currently being updated */ /* returns true if an extra_info field is currently being updated */
static bool extra_info_update_ongoing(void) static bool extra_info_update_ongoing(void)
{ {
const int num_ovls = omap_dss_get_num_overlays(); const int num_mgrs = dss_feat_get_num_mgrs();
struct ovl_priv_data *op;
struct omap_overlay *ovl;
struct mgr_priv_data *mp;
int i; int i;
for (i = 0; i < num_ovls; ++i) { for (i = 0; i < num_mgrs; ++i) {
ovl = omap_dss_get_overlay(i); struct omap_overlay_manager *mgr;
op = get_ovl_priv(ovl); struct omap_overlay *ovl;
struct mgr_priv_data *mp;
if (!ovl->manager)
continue;
mp = get_mgr_priv(ovl->manager); mgr = omap_dss_get_overlay_manager(i);
mp = get_mgr_priv(mgr);
if (!mp->enabled) if (!mp->enabled)
continue; continue;
...@@ -341,9 +356,16 @@ static bool extra_info_update_ongoing(void) ...@@ -341,9 +356,16 @@ static bool extra_info_update_ongoing(void)
if (!mp->updating) if (!mp->updating)
continue; continue;
if (mp->extra_info_dirty || mp->shadow_extra_info_dirty)
return true;
list_for_each_entry(ovl, &mgr->overlays, list) {
struct ovl_priv_data *op = get_ovl_priv(ovl);
if (op->extra_info_dirty || op->shadow_extra_info_dirty) if (op->extra_info_dirty || op->shadow_extra_info_dirty)
return true; return true;
} }
}
return false; return false;
} }
...@@ -601,6 +623,22 @@ static void dss_mgr_write_regs(struct omap_overlay_manager *mgr) ...@@ -601,6 +623,22 @@ static void dss_mgr_write_regs(struct omap_overlay_manager *mgr)
} }
} }
static void dss_mgr_write_regs_extra(struct omap_overlay_manager *mgr)
{
struct mgr_priv_data *mp = get_mgr_priv(mgr);
DSSDBGF("%d", mgr->id);
if (!mp->extra_info_dirty)
return;
dispc_mgr_set_timings(mgr->id, &mp->timings);
mp->extra_info_dirty = false;
if (mp->updating)
mp->shadow_extra_info_dirty = true;
}
static void dss_write_regs_common(void) static void dss_write_regs_common(void)
{ {
const int num_mgrs = omap_dss_get_num_overlay_managers(); const int num_mgrs = omap_dss_get_num_overlay_managers();
...@@ -654,6 +692,7 @@ static void dss_write_regs(void) ...@@ -654,6 +692,7 @@ static void dss_write_regs(void)
} }
dss_mgr_write_regs(mgr); dss_mgr_write_regs(mgr);
dss_mgr_write_regs_extra(mgr);
} }
} }
...@@ -693,6 +732,7 @@ static void mgr_clear_shadow_dirty(struct omap_overlay_manager *mgr) ...@@ -693,6 +732,7 @@ static void mgr_clear_shadow_dirty(struct omap_overlay_manager *mgr)
mp = get_mgr_priv(mgr); mp = get_mgr_priv(mgr);
mp->shadow_info_dirty = false; mp->shadow_info_dirty = false;
mp->shadow_extra_info_dirty = false;
list_for_each_entry(ovl, &mgr->overlays, list) { list_for_each_entry(ovl, &mgr->overlays, list) {
op = get_ovl_priv(ovl); op = get_ovl_priv(ovl);
...@@ -719,6 +759,7 @@ void dss_mgr_start_update(struct omap_overlay_manager *mgr) ...@@ -719,6 +759,7 @@ void dss_mgr_start_update(struct omap_overlay_manager *mgr)
} }
dss_mgr_write_regs(mgr); dss_mgr_write_regs(mgr);
dss_mgr_write_regs_extra(mgr);
dss_write_regs_common(); dss_write_regs_common();
...@@ -1225,6 +1266,35 @@ int dss_mgr_unset_device(struct omap_overlay_manager *mgr) ...@@ -1225,6 +1266,35 @@ int dss_mgr_unset_device(struct omap_overlay_manager *mgr)
return r; return r;
} }
static void dss_apply_mgr_timings(struct omap_overlay_manager *mgr,
struct omap_video_timings *timings)
{
struct mgr_priv_data *mp = get_mgr_priv(mgr);
mp->timings = *timings;
mp->extra_info_dirty = true;
}
void dss_mgr_set_timings(struct omap_overlay_manager *mgr,
struct omap_video_timings *timings)
{
unsigned long flags;
mutex_lock(&apply_lock);
spin_lock_irqsave(&data_lock, flags);
dss_apply_mgr_timings(mgr, timings);
dss_write_regs();
dss_set_go_bits();
spin_unlock_irqrestore(&data_lock, flags);
wait_pending_extra_info_updates();
mutex_unlock(&apply_lock);
}
int dss_ovl_set_info(struct omap_overlay *ovl, int dss_ovl_set_info(struct omap_overlay *ovl,
struct omap_overlay_info *info) struct omap_overlay_info *info)
......
...@@ -177,6 +177,8 @@ void dss_mgr_get_info(struct omap_overlay_manager *mgr, ...@@ -177,6 +177,8 @@ void dss_mgr_get_info(struct omap_overlay_manager *mgr,
int dss_mgr_set_device(struct omap_overlay_manager *mgr, int dss_mgr_set_device(struct omap_overlay_manager *mgr,
struct omap_dss_device *dssdev); struct omap_dss_device *dssdev);
int dss_mgr_unset_device(struct omap_overlay_manager *mgr); int dss_mgr_unset_device(struct omap_overlay_manager *mgr);
void dss_mgr_set_timings(struct omap_overlay_manager *mgr,
struct omap_video_timings *timings);
bool dss_ovl_is_enabled(struct omap_overlay *ovl); bool dss_ovl_is_enabled(struct omap_overlay *ovl);
int dss_ovl_enable(struct omap_overlay *ovl); int dss_ovl_enable(struct omap_overlay *ovl);
......
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