sh_mobile_meram: Add support for NV24 framebuffers

Since the NV24 framebuffer has a CbCr plane that is twice as wide
as the Y plane, it needs to be handled as a special case.

Signed-off-by: Damian Hobson-Garcia <dhobsong@igel.co.jp>
Signed-off-by: Paul Mundt <lethal@linux-sh.org>
This commit is contained in:
Damian 2011-05-18 11:10:08 +00:00 committed by Paul Mundt
parent 7caa4342ca
commit 3fedd2ac76
3 changed files with 28 additions and 9 deletions

View file

@ -627,10 +627,14 @@ static int sh_mobile_lcdc_start(struct sh_mobile_lcdc_priv *priv)
ch->meram_enabled = 0; ch->meram_enabled = 0;
if (ch->info->var.nonstd) if (ch->info->var.nonstd) {
pf = SH_MOBILE_MERAM_PF_NV; if (ch->info->var.bits_per_pixel == 24)
else pf = SH_MOBILE_MERAM_PF_NV24;
else
pf = SH_MOBILE_MERAM_PF_NV;
} else {
pf = SH_MOBILE_MERAM_PF_RGB; pf = SH_MOBILE_MERAM_PF_RGB;
}
ret = mdev->ops->meram_register(mdev, cfg, pitch, ret = mdev->ops->meram_register(mdev, cfg, pitch,
ch->info->var.yres, ch->info->var.yres,

View file

@ -164,6 +164,16 @@ static inline void meram_unmark(struct sh_mobile_meram_priv *priv,
} }
} }
/*
* is this a YCbCr(NV12, NV16 or NV24) colorspace
*/
static inline int is_nvcolor(int cspace)
{
if (cspace == SH_MOBILE_MERAM_PF_NV ||
cspace == SH_MOBILE_MERAM_PF_NV24)
return 1;
return 0;
}
/* /*
* set the next address to fetch * set the next address to fetch
@ -184,7 +194,7 @@ static inline void meram_set_next_addr(struct sh_mobile_meram_priv *priv,
meram_write_icb(priv->base, cfg->icb[0].marker_icb, target, meram_write_icb(priv->base, cfg->icb[0].marker_icb, target,
base_addr_y + cfg->icb[0].cache_unit); base_addr_y + cfg->icb[0].cache_unit);
if (cfg->pixelformat == SH_MOBILE_MERAM_PF_NV) { if (is_nvcolor(cfg->pixelformat)) {
meram_write_icb(priv->base, cfg->icb[1].cache_icb, target, meram_write_icb(priv->base, cfg->icb[1].cache_icb, target,
base_addr_c); base_addr_c);
meram_write_icb(priv->base, cfg->icb[1].marker_icb, target, meram_write_icb(priv->base, cfg->icb[1].marker_icb, target,
@ -208,7 +218,7 @@ static inline void meram_get_next_icb_addr(struct sh_mobile_meram_info *pdata,
icb_offset = 0xc0000000 | (cfg->current_reg << 23); icb_offset = 0xc0000000 | (cfg->current_reg << 23);
*icb_addr_y = icb_offset | (cfg->icb[0].marker_icb << 24); *icb_addr_y = icb_offset | (cfg->icb[0].marker_icb << 24);
if ((*icb_addr_c) && cfg->pixelformat == SH_MOBILE_MERAM_PF_NV) if ((*icb_addr_c) && is_nvcolor(cfg->pixelformat))
*icb_addr_c = icb_offset | (cfg->icb[1].marker_icb << 24); *icb_addr_c = icb_offset | (cfg->icb[1].marker_icb << 24);
} }
@ -316,6 +326,7 @@ static int sh_mobile_meram_register(struct sh_mobile_meram_info *pdata,
return -EINVAL; return -EINVAL;
if (pixelformat != SH_MOBILE_MERAM_PF_NV && if (pixelformat != SH_MOBILE_MERAM_PF_NV &&
pixelformat != SH_MOBILE_MERAM_PF_NV24 &&
pixelformat != SH_MOBILE_MERAM_PF_RGB) pixelformat != SH_MOBILE_MERAM_PF_RGB)
return -EINVAL; return -EINVAL;
@ -366,7 +377,7 @@ static int sh_mobile_meram_register(struct sh_mobile_meram_info *pdata,
n = 2; n = 2;
} }
if (pixelformat == SH_MOBILE_MERAM_PF_NV && n != 2) { if (is_nvcolor(pixelformat) && n != 2) {
dev_err(&pdev->dev, "requires two ICB sets for planar Y/C."); dev_err(&pdev->dev, "requires two ICB sets for planar Y/C.");
error = -EINVAL; error = -EINVAL;
goto err; goto err;
@ -375,7 +386,7 @@ static int sh_mobile_meram_register(struct sh_mobile_meram_info *pdata,
/* we now register the ICB */ /* we now register the ICB */
cfg->pixelformat = pixelformat; cfg->pixelformat = pixelformat;
meram_mark(priv, &cfg->icb[0]); meram_mark(priv, &cfg->icb[0]);
if (pixelformat == SH_MOBILE_MERAM_PF_NV) if (is_nvcolor(pixelformat))
meram_mark(priv, &cfg->icb[1]); meram_mark(priv, &cfg->icb[1]);
/* initialize MERAM */ /* initialize MERAM */
@ -384,6 +395,9 @@ static int sh_mobile_meram_register(struct sh_mobile_meram_info *pdata,
if (pixelformat == SH_MOBILE_MERAM_PF_NV) if (pixelformat == SH_MOBILE_MERAM_PF_NV)
meram_init(priv, &cfg->icb[1], xres, (yres + 1) / 2, meram_init(priv, &cfg->icb[1], xres, (yres + 1) / 2,
&out_pitch); &out_pitch);
else if (pixelformat == SH_MOBILE_MERAM_PF_NV24)
meram_init(priv, &cfg->icb[1], 2 * xres, (yres + 1) / 2,
&out_pitch);
cfg->current_reg = 1; cfg->current_reg = 1;
meram_set_next_addr(priv, cfg, base_addr_y, base_addr_c); meram_set_next_addr(priv, cfg, base_addr_y, base_addr_c);
@ -410,7 +424,7 @@ static int sh_mobile_meram_unregister(struct sh_mobile_meram_info *pdata,
mutex_lock(&priv->lock); mutex_lock(&priv->lock);
/* deinit & unmark */ /* deinit & unmark */
if (cfg->pixelformat == SH_MOBILE_MERAM_PF_NV) { if (is_nvcolor(cfg->pixelformat)) {
meram_deinit(priv, &cfg->icb[1]); meram_deinit(priv, &cfg->icb[1]);
meram_unmark(priv, &cfg->icb[1]); meram_unmark(priv, &cfg->icb[1]);
} }

View file

@ -9,7 +9,8 @@ enum {
enum { enum {
SH_MOBILE_MERAM_PF_NV = 0, SH_MOBILE_MERAM_PF_NV = 0,
SH_MOBILE_MERAM_PF_RGB SH_MOBILE_MERAM_PF_RGB,
SH_MOBILE_MERAM_PF_NV24
}; };