OMAP: DSS2: change manual update scaling setup

Currently the update area on manual update displays is automatically
enlargened to fully cover scaled overlays. This patch makes that
optional, allowing the panel driver to choose if it's used or not.

Signed-off-by: Tomi Valkeinen <tomi.valkeinen@nokia.com>
This commit is contained in:
Tomi Valkeinen 2010-06-09 15:31:34 +03:00
parent ab83b14c82
commit 26a8c2507a
6 changed files with 53 additions and 38 deletions

View File

@ -560,7 +560,8 @@ void omapdss_dsi_vc_enable_hs(int channel, bool enable);
int omapdss_dsi_enable_te(struct omap_dss_device *dssdev, bool enable); int omapdss_dsi_enable_te(struct omap_dss_device *dssdev, bool enable);
int omap_dsi_prepare_update(struct omap_dss_device *dssdev, int omap_dsi_prepare_update(struct omap_dss_device *dssdev,
u16 *x, u16 *y, u16 *w, u16 *h); u16 *x, u16 *y, u16 *w, u16 *h,
bool enlarge_update_area);
int omap_dsi_update(struct omap_dss_device *dssdev, int omap_dsi_update(struct omap_dss_device *dssdev,
int channel, int channel,
u16 x, u16 y, u16 w, u16 h, u16 x, u16 y, u16 w, u16 h,

View File

@ -1135,7 +1135,7 @@ static int taal_update(struct omap_dss_device *dssdev,
goto err; goto err;
} }
r = omap_dsi_prepare_update(dssdev, &x, &y, &w, &h); r = omap_dsi_prepare_update(dssdev, &x, &y, &w, &h, true);
if (r) if (r)
goto err; goto err;

View File

@ -2860,7 +2860,8 @@ static void dsi_framedone_irq_callback(void *data, u32 mask)
} }
int omap_dsi_prepare_update(struct omap_dss_device *dssdev, int omap_dsi_prepare_update(struct omap_dss_device *dssdev,
u16 *x, u16 *y, u16 *w, u16 *h) u16 *x, u16 *y, u16 *w, u16 *h,
bool enlarge_update_area)
{ {
u16 dw, dh; u16 dw, dh;
@ -2884,7 +2885,8 @@ int omap_dsi_prepare_update(struct omap_dss_device *dssdev,
dsi_perf_mark_setup(); dsi_perf_mark_setup();
if (dssdev->manager->caps & OMAP_DSS_OVL_MGR_CAP_DISPC) { if (dssdev->manager->caps & OMAP_DSS_OVL_MGR_CAP_DISPC) {
dss_setup_partial_planes(dssdev, x, y, w, h); dss_setup_partial_planes(dssdev, x, y, w, h,
enlarge_update_area);
dispc_set_lcd_size(*w, *h); dispc_set_lcd_size(*w, *h);
} }

View File

@ -199,7 +199,8 @@ int dss_init_overlay_managers(struct platform_device *pdev);
void dss_uninit_overlay_managers(struct platform_device *pdev); void dss_uninit_overlay_managers(struct platform_device *pdev);
int dss_mgr_wait_for_go_ovl(struct omap_overlay *ovl); int dss_mgr_wait_for_go_ovl(struct omap_overlay *ovl);
void dss_setup_partial_planes(struct omap_dss_device *dssdev, void dss_setup_partial_planes(struct omap_dss_device *dssdev,
u16 *x, u16 *y, u16 *w, u16 *h); u16 *x, u16 *y, u16 *w, u16 *h,
bool enlarge_update_area);
void dss_start_update(struct omap_dss_device *dssdev); void dss_start_update(struct omap_dss_device *dssdev);
/* overlay */ /* overlay */

View File

@ -440,6 +440,10 @@ struct manager_cache_data {
/* manual update region */ /* manual update region */
u16 x, y, w, h; u16 x, y, w, h;
/* enlarge the update area if the update area contains scaled
* overlays */
bool enlarge_update_area;
}; };
static struct { static struct {
@ -721,6 +725,7 @@ static int configure_overlay(enum omap_plane plane)
u16 x, y, w, h; u16 x, y, w, h;
u32 paddr; u32 paddr;
int r; int r;
u16 orig_w, orig_h, orig_outw, orig_outh;
DSSDBGF("%d", plane); DSSDBGF("%d", plane);
@ -741,8 +746,16 @@ static int configure_overlay(enum omap_plane plane)
outh = c->out_height == 0 ? c->height : c->out_height; outh = c->out_height == 0 ? c->height : c->out_height;
paddr = c->paddr; paddr = c->paddr;
orig_w = w;
orig_h = h;
orig_outw = outw;
orig_outh = outh;
if (c->manual_update && mc->do_manual_update) { if (c->manual_update && mc->do_manual_update) {
unsigned bpp; unsigned bpp;
unsigned scale_x_m = w, scale_x_d = outw;
unsigned scale_y_m = h, scale_y_d = outh;
/* If the overlay is outside the update region, disable it */ /* If the overlay is outside the update region, disable it */
if (!rectangle_intersects(mc->x, mc->y, mc->w, mc->h, if (!rectangle_intersects(mc->x, mc->y, mc->w, mc->h,
x, y, outw, outh)) { x, y, outw, outh)) {
@ -773,39 +786,33 @@ static int configure_overlay(enum omap_plane plane)
BUG(); BUG();
} }
if (dispc_is_overlay_scaled(c)) { if (mc->x > c->pos_x) {
/* If the overlay is scaled, the update area has x = 0;
* already been enlarged to cover the whole overlay. We outw -= (mc->x - c->pos_x);
* only need to adjust x/y here */ paddr += (mc->x - c->pos_x) *
x = c->pos_x - mc->x; scale_x_m / scale_x_d * bpp / 8;
y = c->pos_y - mc->y;
} else { } else {
if (mc->x > c->pos_x) { x = c->pos_x - mc->x;
x = 0;
w -= (mc->x - c->pos_x);
paddr += (mc->x - c->pos_x) * bpp / 8;
} else {
x = c->pos_x - mc->x;
}
if (mc->y > c->pos_y) {
y = 0;
h -= (mc->y - c->pos_y);
paddr += (mc->y - c->pos_y) * c->screen_width *
bpp / 8;
} else {
y = c->pos_y - mc->y;
}
if (mc->w < (x+w))
w -= (x+w) - (mc->w);
if (mc->h < (y+h))
h -= (y+h) - (mc->h);
outw = w;
outh = h;
} }
if (mc->y > c->pos_y) {
y = 0;
outh -= (mc->y - c->pos_y);
paddr += (mc->y - c->pos_y) *
scale_y_m / scale_y_d *
c->screen_width * bpp / 8;
} else {
y = c->pos_y - mc->y;
}
if (mc->w < (x + outw))
outw -= (x + outw) - (mc->w);
if (mc->h < (y + outh))
outh -= (y + outh) - (mc->h);
w = w * outw / orig_outw;
h = h * outh / orig_outh;
} }
r = dispc_setup_plane(plane, r = dispc_setup_plane(plane,
@ -963,7 +970,7 @@ static void make_even(u16 *x, u16 *w)
/* Configure dispc for partial update. Return possibly modified update /* Configure dispc for partial update. Return possibly modified update
* area */ * area */
void dss_setup_partial_planes(struct omap_dss_device *dssdev, void dss_setup_partial_planes(struct omap_dss_device *dssdev,
u16 *xi, u16 *yi, u16 *wi, u16 *hi) u16 *xi, u16 *yi, u16 *wi, u16 *hi, bool enlarge_update_area)
{ {
struct overlay_cache_data *oc; struct overlay_cache_data *oc;
struct manager_cache_data *mc; struct manager_cache_data *mc;
@ -1015,6 +1022,9 @@ void dss_setup_partial_planes(struct omap_dss_device *dssdev,
oc->dirty = true; oc->dirty = true;
if (!enlarge_update_area)
continue;
if (!oc->enabled) if (!oc->enabled)
continue; continue;
@ -1074,6 +1084,7 @@ void dss_setup_partial_planes(struct omap_dss_device *dssdev,
mc = &dss_cache.manager_cache[mgr->id]; mc = &dss_cache.manager_cache[mgr->id];
mc->do_manual_update = true; mc->do_manual_update = true;
mc->enlarge_update_area = enlarge_update_area;
mc->x = x; mc->x = x;
mc->y = y; mc->y = y;
mc->w = w; mc->w = w;

View File

@ -886,7 +886,7 @@ int omap_rfbi_prepare_update(struct omap_dss_device *dssdev,
return -EINVAL; return -EINVAL;
if (dssdev->manager->caps & OMAP_DSS_OVL_MGR_CAP_DISPC) { if (dssdev->manager->caps & OMAP_DSS_OVL_MGR_CAP_DISPC) {
dss_setup_partial_planes(dssdev, x, y, w, h); dss_setup_partial_planes(dssdev, x, y, w, h, true);
dispc_set_lcd_size(*w, *h); dispc_set_lcd_size(*w, *h);
} }