mirror of
https://github.com/followmsi/android_kernel_google_msm.git
synced 2024-11-06 23:17:41 +00:00
04e554807c
Change-Id: Ibead64ce2e901dede2ddd1b86088b88f2350ce92 Signed-off-by: Duy Truong <dtruong@codeaurora.org>
730 lines
18 KiB
C
730 lines
18 KiB
C
/* Copyright (c) 2009-2012, The Linux Foundation. All rights reserved.
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License version 2 and
|
|
* only version 2 as published by the Free Software Foundation.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/sched.h>
|
|
#include <linux/time.h>
|
|
#include <linux/init.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/hrtimer.h>
|
|
#include <linux/delay.h>
|
|
#include <mach/hardware.h>
|
|
#include <linux/io.h>
|
|
|
|
#include <asm/system.h>
|
|
#include <asm/mach-types.h>
|
|
#include <linux/semaphore.h>
|
|
#include <linux/spinlock.h>
|
|
|
|
#include <linux/fb.h>
|
|
|
|
#include "mdp.h"
|
|
#include "msm_fb.h"
|
|
#include "mdp4.h"
|
|
|
|
static struct mdp4_overlay_pipe *mddi_pipe;
|
|
static struct msm_fb_data_type *mddi_mfd;
|
|
static int busy_wait_cnt;
|
|
|
|
static int vsync_start_y_adjust = 4;
|
|
|
|
static int dmap_vsync_enable;
|
|
|
|
void mdp_dmap_vsync_set(int enable)
|
|
{
|
|
dmap_vsync_enable = enable;
|
|
}
|
|
|
|
int mdp_dmap_vsync_get(void)
|
|
{
|
|
return dmap_vsync_enable;
|
|
}
|
|
|
|
void mdp4_mddi_vsync_enable(struct msm_fb_data_type *mfd,
|
|
struct mdp4_overlay_pipe *pipe, int which)
|
|
{
|
|
uint32 start_y, data, tear_en;
|
|
|
|
tear_en = (1 << which);
|
|
|
|
if ((mfd->use_mdp_vsync) && (mfd->ibuf.vsync_enable) &&
|
|
(mfd->panel_info.lcd.vsync_enable)) {
|
|
|
|
if (mdp_hw_revision < MDP4_REVISION_V2_1) {
|
|
/* need dmas dmap switch */
|
|
if (which == 0 && dmap_vsync_enable == 0 &&
|
|
mfd->panel_info.lcd.rev < 2) /* dma_p */
|
|
return;
|
|
}
|
|
|
|
if (vsync_start_y_adjust <= pipe->dst_y)
|
|
start_y = pipe->dst_y - vsync_start_y_adjust;
|
|
else
|
|
start_y = (mfd->total_lcd_lines - 1) -
|
|
(vsync_start_y_adjust - pipe->dst_y);
|
|
if (which == 0)
|
|
MDP_OUTP(MDP_BASE + 0x210, start_y); /* primary */
|
|
else
|
|
MDP_OUTP(MDP_BASE + 0x214, start_y); /* secondary */
|
|
|
|
data = inpdw(MDP_BASE + 0x20c);
|
|
data |= tear_en;
|
|
MDP_OUTP(MDP_BASE + 0x20c, data);
|
|
} else {
|
|
data = inpdw(MDP_BASE + 0x20c);
|
|
data &= ~tear_en;
|
|
MDP_OUTP(MDP_BASE + 0x20c, data);
|
|
}
|
|
}
|
|
|
|
#define WHOLESCREEN
|
|
|
|
void mdp4_overlay_update_lcd(struct msm_fb_data_type *mfd)
|
|
{
|
|
MDPIBUF *iBuf = &mfd->ibuf;
|
|
uint8 *src;
|
|
int ptype;
|
|
uint32 mddi_ld_param;
|
|
uint16 mddi_vdo_packet_reg;
|
|
struct mdp4_overlay_pipe *pipe;
|
|
int ret;
|
|
|
|
if (mfd->key != MFD_KEY)
|
|
return;
|
|
|
|
mddi_mfd = mfd; /* keep it */
|
|
|
|
/* MDP cmd block enable */
|
|
mdp_pipe_ctrl(MDP_CMD_BLOCK, MDP_BLOCK_POWER_ON, FALSE);
|
|
|
|
if (mddi_pipe == NULL) {
|
|
ptype = mdp4_overlay_format2type(mfd->fb_imgType);
|
|
if (ptype < 0)
|
|
printk(KERN_INFO "%s: format2type failed\n", __func__);
|
|
pipe = mdp4_overlay_pipe_alloc(ptype, MDP4_MIXER0);
|
|
if (pipe == NULL)
|
|
printk(KERN_INFO "%s: pipe_alloc failed\n", __func__);
|
|
pipe->pipe_used++;
|
|
pipe->mixer_num = MDP4_MIXER0;
|
|
pipe->src_format = mfd->fb_imgType;
|
|
mdp4_overlay_panel_mode(pipe->mixer_num, MDP4_PANEL_MDDI);
|
|
ret = mdp4_overlay_format2pipe(pipe);
|
|
if (ret < 0)
|
|
printk(KERN_INFO "%s: format2type failed\n", __func__);
|
|
|
|
mddi_pipe = pipe; /* keep it */
|
|
mddi_ld_param = 0;
|
|
mddi_vdo_packet_reg = mfd->panel_info.mddi.vdopkt;
|
|
|
|
if (mdp_hw_revision == MDP4_REVISION_V2_1) {
|
|
uint32 data;
|
|
|
|
data = inpdw(MDP_BASE + 0x0028);
|
|
data &= ~0x0300; /* bit 8, 9, MASTER4 */
|
|
if (mfd->fbi->var.xres == 540) /* qHD, 540x960 */
|
|
data |= 0x0200;
|
|
else
|
|
data |= 0x0100;
|
|
|
|
MDP_OUTP(MDP_BASE + 0x00028, data);
|
|
}
|
|
|
|
if (mfd->panel_info.type == MDDI_PANEL) {
|
|
if (mfd->panel_info.pdest == DISPLAY_1)
|
|
mddi_ld_param = 0;
|
|
else
|
|
mddi_ld_param = 1;
|
|
} else {
|
|
mddi_ld_param = 2;
|
|
}
|
|
|
|
MDP_OUTP(MDP_BASE + 0x00090, mddi_ld_param);
|
|
|
|
if (mfd->panel_info.bpp == 24)
|
|
MDP_OUTP(MDP_BASE + 0x00094,
|
|
(MDDI_VDO_PACKET_DESC_24 << 16) | mddi_vdo_packet_reg);
|
|
else if (mfd->panel_info.bpp == 16)
|
|
MDP_OUTP(MDP_BASE + 0x00094,
|
|
(MDDI_VDO_PACKET_DESC_16 << 16) | mddi_vdo_packet_reg);
|
|
else
|
|
MDP_OUTP(MDP_BASE + 0x00094,
|
|
(MDDI_VDO_PACKET_DESC << 16) | mddi_vdo_packet_reg);
|
|
|
|
MDP_OUTP(MDP_BASE + 0x00098, 0x01);
|
|
mdp4_init_writeback_buf(mfd, MDP4_MIXER0);
|
|
pipe->ov_blt_addr = 0;
|
|
pipe->dma_blt_addr = 0;
|
|
} else {
|
|
pipe = mddi_pipe;
|
|
}
|
|
|
|
/* 0 for dma_p, client_id = 0 */
|
|
MDP_OUTP(MDP_BASE + 0x00090, 0);
|
|
|
|
|
|
src = (uint8 *) iBuf->buf;
|
|
|
|
#ifdef WHOLESCREEN
|
|
|
|
{
|
|
struct fb_info *fbi;
|
|
|
|
fbi = mfd->fbi;
|
|
pipe->src_height = fbi->var.yres;
|
|
pipe->src_width = fbi->var.xres;
|
|
pipe->src_h = fbi->var.yres;
|
|
pipe->src_w = fbi->var.xres;
|
|
pipe->src_y = 0;
|
|
pipe->src_x = 0;
|
|
pipe->dst_h = fbi->var.yres;
|
|
pipe->dst_w = fbi->var.xres;
|
|
pipe->dst_y = 0;
|
|
pipe->dst_x = 0;
|
|
pipe->srcp0_addr = (uint32)src;
|
|
pipe->srcp0_ystride = fbi->fix.line_length;
|
|
}
|
|
|
|
#else
|
|
if (mdp4_overlay_active(MDP4_MIXER0)) {
|
|
struct fb_info *fbi;
|
|
|
|
fbi = mfd->fbi;
|
|
pipe->src_height = fbi->var.yres;
|
|
pipe->src_width = fbi->var.xres;
|
|
pipe->src_h = fbi->var.yres;
|
|
pipe->src_w = fbi->var.xres;
|
|
pipe->src_y = 0;
|
|
pipe->src_x = 0;
|
|
pipe->dst_h = fbi->var.yres;
|
|
pipe->dst_w = fbi->var.xres;
|
|
pipe->dst_y = 0;
|
|
pipe->dst_x = 0;
|
|
pipe->srcp0_addr = (uint32) src;
|
|
pipe->srcp0_ystride = fbi->fix.line_length;
|
|
} else {
|
|
/* starting input address */
|
|
src += (iBuf->dma_x + iBuf->dma_y * iBuf->ibuf_width)
|
|
* iBuf->bpp;
|
|
|
|
pipe->src_height = iBuf->dma_h;
|
|
pipe->src_width = iBuf->dma_w;
|
|
pipe->src_h = iBuf->dma_h;
|
|
pipe->src_w = iBuf->dma_w;
|
|
pipe->src_y = 0;
|
|
pipe->src_x = 0;
|
|
pipe->dst_h = iBuf->dma_h;
|
|
pipe->dst_w = iBuf->dma_w;
|
|
pipe->dst_y = iBuf->dma_y;
|
|
pipe->dst_x = iBuf->dma_x;
|
|
pipe->srcp0_addr = (uint32) src;
|
|
pipe->srcp0_ystride = iBuf->ibuf_width * iBuf->bpp;
|
|
}
|
|
#endif
|
|
|
|
pipe->mixer_stage = MDP4_MIXER_STAGE_BASE;
|
|
|
|
mdp4_overlay_rgb_setup(pipe);
|
|
|
|
mdp4_mixer_stage_up(pipe, 1);
|
|
|
|
mdp4_overlayproc_cfg(pipe);
|
|
|
|
mdp4_overlay_dmap_xy(pipe);
|
|
|
|
mdp4_overlay_dmap_cfg(mfd, 0);
|
|
mdp4_mixer_stage_commit(pipe->mixer_num);
|
|
mdp4_mddi_vsync_enable(mfd, pipe, 0);
|
|
|
|
/* MDP cmd block disable */
|
|
mdp_pipe_ctrl(MDP_CMD_BLOCK, MDP_BLOCK_POWER_OFF, FALSE);
|
|
}
|
|
|
|
int mdp4_mddi_overlay_blt_start(struct msm_fb_data_type *mfd)
|
|
{
|
|
unsigned long flag;
|
|
|
|
pr_debug("%s: blt_end=%d blt_addr=%x pid=%d\n",
|
|
__func__, mddi_pipe->blt_end,
|
|
(int)mddi_pipe->ov_blt_addr, current->pid);
|
|
|
|
mdp4_allocate_writeback_buf(mfd, MDP4_MIXER0);
|
|
|
|
if (mfd->ov0_wb_buf->write_addr == 0) {
|
|
pr_info("%s: no blt_base assigned\n", __func__);
|
|
return -EBUSY;
|
|
}
|
|
|
|
if (mddi_pipe->ov_blt_addr == 0) {
|
|
mdp4_mddi_dma_busy_wait(mfd);
|
|
spin_lock_irqsave(&mdp_spin_lock, flag);
|
|
mddi_pipe->blt_end = 0;
|
|
mddi_pipe->blt_cnt = 0;
|
|
mddi_pipe->ov_cnt = 0;
|
|
mddi_pipe->dmap_cnt = 0;
|
|
mddi_pipe->ov_blt_addr = mfd->ov0_wb_buf->write_addr;
|
|
mddi_pipe->dma_blt_addr = mfd->ov0_wb_buf->write_addr;
|
|
mdp4_stat.blt_mddi++;
|
|
spin_unlock_irqrestore(&mdp_spin_lock, flag);
|
|
return 0;
|
|
}
|
|
|
|
return -EBUSY;
|
|
}
|
|
|
|
int mdp4_mddi_overlay_blt_stop(struct msm_fb_data_type *mfd)
|
|
{
|
|
unsigned long flag;
|
|
|
|
pr_debug("%s: blt_end=%d blt_addr=%x\n",
|
|
__func__, mddi_pipe->blt_end, (int)mddi_pipe->ov_blt_addr);
|
|
|
|
if ((mddi_pipe->blt_end == 0) && mddi_pipe->ov_blt_addr) {
|
|
spin_lock_irqsave(&mdp_spin_lock, flag);
|
|
mddi_pipe->blt_end = 1; /* mark as end */
|
|
spin_unlock_irqrestore(&mdp_spin_lock, flag);
|
|
return 0;
|
|
}
|
|
|
|
return -EBUSY;
|
|
}
|
|
|
|
int mdp4_mddi_overlay_blt_offset(struct msm_fb_data_type *mfd,
|
|
struct msmfb_overlay_blt *req)
|
|
{
|
|
req->offset = 0;
|
|
req->width = mddi_pipe->src_width;
|
|
req->height = mddi_pipe->src_height;
|
|
req->bpp = mddi_pipe->bpp;
|
|
|
|
return sizeof(*req);
|
|
}
|
|
|
|
void mdp4_mddi_overlay_blt(struct msm_fb_data_type *mfd,
|
|
struct msmfb_overlay_blt *req)
|
|
{
|
|
if (req->enable)
|
|
mdp4_mddi_overlay_blt_start(mfd);
|
|
else if (req->enable == 0)
|
|
mdp4_mddi_overlay_blt_stop(mfd);
|
|
|
|
}
|
|
|
|
void mdp4_blt_xy_update(struct mdp4_overlay_pipe *pipe)
|
|
{
|
|
uint32 off, addr, addr2;
|
|
int bpp;
|
|
char *overlay_base;
|
|
|
|
if (pipe->ov_blt_addr == 0)
|
|
return;
|
|
|
|
|
|
#ifdef BLT_RGB565
|
|
bpp = 2; /* overlay ouput is RGB565 */
|
|
#else
|
|
bpp = 3; /* overlay ouput is RGB888 */
|
|
#endif
|
|
off = 0;
|
|
if (pipe->dmap_cnt & 0x01)
|
|
off = pipe->src_height * pipe->src_width * bpp;
|
|
|
|
addr = pipe->ov_blt_addr + off;
|
|
|
|
/* dmap */
|
|
MDP_OUTP(MDP_BASE + 0x90008, addr);
|
|
|
|
off = 0;
|
|
if (pipe->ov_cnt & 0x01)
|
|
off = pipe->src_height * pipe->src_width * bpp;
|
|
addr2 = pipe->ov_blt_addr + off;
|
|
/* overlay 0 */
|
|
overlay_base = MDP_BASE + MDP4_OVERLAYPROC0_BASE;/* 0x10000 */
|
|
outpdw(overlay_base + 0x000c, addr2);
|
|
outpdw(overlay_base + 0x001c, addr2);
|
|
}
|
|
|
|
void mdp4_primary_rdptr(void)
|
|
{
|
|
}
|
|
|
|
/*
|
|
* mdp4_dmap_done_mddi: called from isr
|
|
*/
|
|
void mdp4_dma_p_done_mddi(struct mdp_dma_data *dma)
|
|
{
|
|
int diff;
|
|
|
|
mddi_pipe->dmap_cnt++;
|
|
diff = mddi_pipe->ov_cnt - mddi_pipe->dmap_cnt;
|
|
pr_debug("%s: ov_cnt=%d dmap_cnt=%d\n",
|
|
__func__, mddi_pipe->ov_cnt, mddi_pipe->dmap_cnt);
|
|
|
|
if (diff <= 0) {
|
|
spin_lock(&mdp_spin_lock);
|
|
dma->dmap_busy = FALSE;
|
|
complete(&dma->dmap_comp);
|
|
spin_unlock(&mdp_spin_lock);
|
|
|
|
if (mddi_pipe->blt_end) {
|
|
mddi_pipe->blt_end = 0;
|
|
mddi_pipe->ov_blt_addr = 0;
|
|
mddi_pipe->dma_blt_addr = 0;
|
|
pr_debug("%s: END, ov_cnt=%d dmap_cnt=%d\n", __func__,
|
|
mddi_pipe->ov_cnt, mddi_pipe->dmap_cnt);
|
|
mdp_intr_mask &= ~INTR_DMA_P_DONE;
|
|
outp32(MDP_INTR_ENABLE, mdp_intr_mask);
|
|
}
|
|
|
|
mdp_pipe_ctrl(MDP_OVERLAY0_BLOCK, MDP_BLOCK_POWER_OFF, TRUE);
|
|
mdp_disable_irq_nosync(MDP_DMA2_TERM); /* disable intr */
|
|
return;
|
|
}
|
|
|
|
spin_lock(&mdp_spin_lock);
|
|
dma->busy = FALSE;
|
|
spin_unlock(&mdp_spin_lock);
|
|
complete(&dma->comp);
|
|
if (busy_wait_cnt)
|
|
busy_wait_cnt--;
|
|
|
|
pr_debug("%s: kickoff dmap\n", __func__);
|
|
|
|
mdp4_blt_xy_update(mddi_pipe);
|
|
/* kick off dmap */
|
|
outpdw(MDP_BASE + 0x000c, 0x0);
|
|
mdp4_stat.kickoff_dmap++;
|
|
mdp_pipe_ctrl(MDP_OVERLAY0_BLOCK, MDP_BLOCK_POWER_OFF, TRUE);
|
|
}
|
|
|
|
/*
|
|
* mdp4_overlay0_done_mddi: called from isr
|
|
*/
|
|
void mdp4_overlay0_done_mddi(struct mdp_dma_data *dma)
|
|
{
|
|
int diff;
|
|
|
|
if (mddi_pipe->ov_blt_addr == 0) {
|
|
mdp_pipe_ctrl(MDP_OVERLAY0_BLOCK, MDP_BLOCK_POWER_OFF, TRUE);
|
|
spin_lock(&mdp_spin_lock);
|
|
dma->busy = FALSE;
|
|
spin_unlock(&mdp_spin_lock);
|
|
complete(&dma->comp);
|
|
|
|
if (busy_wait_cnt)
|
|
busy_wait_cnt--;
|
|
mdp_disable_irq_nosync(MDP_OVERLAY0_TERM);
|
|
|
|
return;
|
|
}
|
|
|
|
/* blt enabled */
|
|
if (mddi_pipe->blt_end == 0)
|
|
mddi_pipe->ov_cnt++;
|
|
|
|
pr_debug("%s: ov_cnt=%d dmap_cnt=%d\n",
|
|
__func__, mddi_pipe->ov_cnt, mddi_pipe->dmap_cnt);
|
|
|
|
if (mddi_pipe->blt_cnt == 0) {
|
|
/* first kickoff since blt enabled */
|
|
mdp_intr_mask |= INTR_DMA_P_DONE;
|
|
outp32(MDP_INTR_ENABLE, mdp_intr_mask);
|
|
}
|
|
|
|
mddi_pipe->blt_cnt++;
|
|
|
|
diff = mddi_pipe->ov_cnt - mddi_pipe->dmap_cnt;
|
|
if (diff >= 2) {
|
|
mdp_disable_irq_nosync(MDP_OVERLAY0_TERM);
|
|
return;
|
|
}
|
|
|
|
spin_lock(&mdp_spin_lock);
|
|
dma->busy = FALSE;
|
|
dma->dmap_busy = TRUE;
|
|
spin_unlock(&mdp_spin_lock);
|
|
complete(&dma->comp);
|
|
|
|
if (busy_wait_cnt)
|
|
busy_wait_cnt--;
|
|
|
|
pr_debug("%s: kickoff dmap\n", __func__);
|
|
|
|
mdp4_blt_xy_update(mddi_pipe);
|
|
mdp_enable_irq(MDP_DMA2_TERM); /* enable intr */
|
|
/* kick off dmap */
|
|
outpdw(MDP_BASE + 0x000c, 0x0);
|
|
mdp4_stat.kickoff_dmap++;
|
|
mdp_disable_irq_nosync(MDP_OVERLAY0_TERM);
|
|
}
|
|
|
|
void mdp4_mddi_overlay_restore(void)
|
|
{
|
|
if (mddi_mfd == NULL)
|
|
return;
|
|
|
|
pr_debug("%s: resotre, pid=%d\n", __func__, current->pid);
|
|
|
|
if (mddi_mfd->panel_power_on == 0)
|
|
return;
|
|
if (mddi_mfd && mddi_pipe) {
|
|
mdp4_mddi_dma_busy_wait(mddi_mfd);
|
|
mdp4_overlay_update_lcd(mddi_mfd);
|
|
|
|
if (mddi_pipe->ov_blt_addr)
|
|
mdp4_mddi_blt_dmap_busy_wait(mddi_mfd);
|
|
mdp4_mddi_overlay_kickoff(mddi_mfd, mddi_pipe);
|
|
mddi_mfd->dma_update_flag = 1;
|
|
}
|
|
if (mdp_hw_revision < MDP4_REVISION_V2_1) /* need dmas dmap switch */
|
|
mdp4_mddi_overlay_dmas_restore();
|
|
}
|
|
|
|
void mdp4_mddi_blt_dmap_busy_wait(struct msm_fb_data_type *mfd)
|
|
{
|
|
unsigned long flag;
|
|
int need_wait = 0;
|
|
|
|
spin_lock_irqsave(&mdp_spin_lock, flag);
|
|
if (mfd->dma->dmap_busy == TRUE) {
|
|
INIT_COMPLETION(mfd->dma->dmap_comp);
|
|
need_wait++;
|
|
}
|
|
spin_unlock_irqrestore(&mdp_spin_lock, flag);
|
|
|
|
if (need_wait) {
|
|
/* wait until DMA finishes the current job */
|
|
wait_for_completion(&mfd->dma->dmap_comp);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* mdp4_mddi_cmd_dma_busy_wait: check mddi link activity
|
|
* mddi link is a shared resource and it can only be used
|
|
* while it is in idle state.
|
|
* ov_mutex need to be acquired before call this function.
|
|
*/
|
|
void mdp4_mddi_dma_busy_wait(struct msm_fb_data_type *mfd)
|
|
{
|
|
unsigned long flag;
|
|
int need_wait = 0;
|
|
|
|
pr_debug("%s: START, pid=%d\n", __func__, current->pid);
|
|
spin_lock_irqsave(&mdp_spin_lock, flag);
|
|
if (mfd->dma->busy == TRUE) {
|
|
if (busy_wait_cnt == 0)
|
|
INIT_COMPLETION(mfd->dma->comp);
|
|
busy_wait_cnt++;
|
|
need_wait++;
|
|
}
|
|
spin_unlock_irqrestore(&mdp_spin_lock, flag);
|
|
|
|
|
|
if (need_wait) {
|
|
/* wait until DMA finishes the current job */
|
|
pr_debug("%s: PENDING, pid=%d\n", __func__, current->pid);
|
|
wait_for_completion(&mfd->dma->comp);
|
|
}
|
|
pr_debug("%s: DONE, pid=%d\n", __func__, current->pid);
|
|
}
|
|
|
|
void mdp4_mddi_kickoff_video(struct msm_fb_data_type *mfd,
|
|
struct mdp4_overlay_pipe *pipe)
|
|
{
|
|
/*
|
|
* a video kickoff may happen before UI kickoff after
|
|
* blt enabled. mdp4_overlay_update_lcd() need
|
|
* to be called before kickoff.
|
|
* vice versa for blt disabled.
|
|
*/
|
|
if (mddi_pipe->ov_blt_addr && mddi_pipe->blt_cnt == 0)
|
|
mdp4_overlay_update_lcd(mfd); /* first time */
|
|
else if (mddi_pipe->ov_blt_addr == 0 && mddi_pipe->blt_cnt) {
|
|
mdp4_overlay_update_lcd(mfd); /* last time */
|
|
mddi_pipe->blt_cnt = 0;
|
|
}
|
|
|
|
pr_debug("%s: blt_addr=%d blt_cnt=%d\n",
|
|
__func__, (int)mddi_pipe->ov_blt_addr, mddi_pipe->blt_cnt);
|
|
|
|
if (mddi_pipe->ov_blt_addr)
|
|
mdp4_mddi_blt_dmap_busy_wait(mddi_mfd);
|
|
mdp4_mddi_overlay_kickoff(mfd, pipe);
|
|
}
|
|
|
|
void mdp4_mddi_kickoff_ui(struct msm_fb_data_type *mfd,
|
|
struct mdp4_overlay_pipe *pipe)
|
|
{
|
|
pr_debug("%s: pid=%d\n", __func__, current->pid);
|
|
mdp4_mddi_overlay_kickoff(mfd, pipe);
|
|
}
|
|
|
|
|
|
void mdp4_mddi_overlay_kickoff(struct msm_fb_data_type *mfd,
|
|
struct mdp4_overlay_pipe *pipe)
|
|
{
|
|
unsigned long flag;
|
|
|
|
mdp_enable_irq(MDP_OVERLAY0_TERM);
|
|
spin_lock_irqsave(&mdp_spin_lock, flag);
|
|
mfd->dma->busy = TRUE;
|
|
if (mddi_pipe->ov_blt_addr)
|
|
mfd->dma->dmap_busy = TRUE;
|
|
spin_unlock_irqrestore(&mdp_spin_lock, flag);
|
|
/* start OVERLAY pipe */
|
|
mdp_pipe_kickoff(MDP_OVERLAY0_TERM, mfd);
|
|
mdp4_stat.kickoff_ov0++;
|
|
}
|
|
|
|
void mdp4_dma_s_update_lcd(struct msm_fb_data_type *mfd,
|
|
struct mdp4_overlay_pipe *pipe)
|
|
{
|
|
MDPIBUF *iBuf = &mfd->ibuf;
|
|
uint32 outBpp = iBuf->bpp;
|
|
uint16 mddi_vdo_packet_reg;
|
|
uint32 dma_s_cfg_reg;
|
|
|
|
dma_s_cfg_reg = 0;
|
|
|
|
if (mfd->fb_imgType == MDP_RGBA_8888)
|
|
dma_s_cfg_reg |= DMA_PACK_PATTERN_BGR; /* on purpose */
|
|
else if (mfd->fb_imgType == MDP_BGR_565)
|
|
dma_s_cfg_reg |= DMA_PACK_PATTERN_BGR;
|
|
else
|
|
dma_s_cfg_reg |= DMA_PACK_PATTERN_RGB;
|
|
|
|
if (outBpp == 4)
|
|
dma_s_cfg_reg |= (1 << 26); /* xRGB8888 */
|
|
else if (outBpp == 2)
|
|
dma_s_cfg_reg |= DMA_IBUF_FORMAT_RGB565;
|
|
|
|
dma_s_cfg_reg |= DMA_DITHER_EN;
|
|
|
|
/* MDP cmd block enable */
|
|
mdp_pipe_ctrl(MDP_CMD_BLOCK, MDP_BLOCK_POWER_ON, FALSE);
|
|
/* PIXELSIZE */
|
|
MDP_OUTP(MDP_BASE + 0xa0004, (pipe->dst_h << 16 | pipe->dst_w));
|
|
MDP_OUTP(MDP_BASE + 0xa0008, pipe->srcp0_addr); /* ibuf address */
|
|
MDP_OUTP(MDP_BASE + 0xa000c, pipe->srcp0_ystride);/* ystride */
|
|
|
|
if (mfd->panel_info.bpp == 24) {
|
|
dma_s_cfg_reg |= DMA_DSTC0G_8BITS | /* 666 18BPP */
|
|
DMA_DSTC1B_8BITS | DMA_DSTC2R_8BITS;
|
|
} else if (mfd->panel_info.bpp == 18) {
|
|
dma_s_cfg_reg |= DMA_DSTC0G_6BITS | /* 666 18BPP */
|
|
DMA_DSTC1B_6BITS | DMA_DSTC2R_6BITS;
|
|
} else {
|
|
dma_s_cfg_reg |= DMA_DSTC0G_6BITS | /* 565 16BPP */
|
|
DMA_DSTC1B_5BITS | DMA_DSTC2R_5BITS;
|
|
}
|
|
|
|
MDP_OUTP(MDP_BASE + 0xa0010, (pipe->dst_y << 16) | pipe->dst_x);
|
|
|
|
/* 1 for dma_s, client_id = 0 */
|
|
MDP_OUTP(MDP_BASE + 0x00090, 1);
|
|
|
|
mddi_vdo_packet_reg = mfd->panel_info.mddi.vdopkt;
|
|
|
|
if (mfd->panel_info.bpp == 24)
|
|
MDP_OUTP(MDP_BASE + 0x00094,
|
|
(MDDI_VDO_PACKET_DESC_24 << 16) | mddi_vdo_packet_reg);
|
|
else if (mfd->panel_info.bpp == 16)
|
|
MDP_OUTP(MDP_BASE + 0x00094,
|
|
(MDDI_VDO_PACKET_DESC_16 << 16) | mddi_vdo_packet_reg);
|
|
else
|
|
MDP_OUTP(MDP_BASE + 0x00094,
|
|
(MDDI_VDO_PACKET_DESC << 16) | mddi_vdo_packet_reg);
|
|
|
|
MDP_OUTP(MDP_BASE + 0x00098, 0x01);
|
|
|
|
MDP_OUTP(MDP_BASE + 0xa0000, dma_s_cfg_reg);
|
|
|
|
mdp4_mddi_vsync_enable(mfd, pipe, 1);
|
|
|
|
/* MDP cmd block disable */
|
|
mdp_pipe_ctrl(MDP_CMD_BLOCK, MDP_BLOCK_POWER_OFF, FALSE);
|
|
}
|
|
|
|
void mdp4_mddi_dma_s_kickoff(struct msm_fb_data_type *mfd,
|
|
struct mdp4_overlay_pipe *pipe)
|
|
{
|
|
mdp_enable_irq(MDP_DMA_S_TERM);
|
|
|
|
if (mddi_pipe->ov_blt_addr == 0)
|
|
mfd->dma->busy = TRUE;
|
|
|
|
mfd->ibuf_flushed = TRUE;
|
|
/* start dma_s pipe */
|
|
mdp_pipe_kickoff(MDP_DMA_S_TERM, mfd);
|
|
mdp4_stat.kickoff_dmas++;
|
|
|
|
/* wait until DMA finishes the current job */
|
|
wait_for_completion(&mfd->dma->comp);
|
|
mdp_disable_irq(MDP_DMA_S_TERM);
|
|
}
|
|
|
|
void mdp4_mddi_overlay_dmas_restore(void)
|
|
{
|
|
/* mutex held by caller */
|
|
if (mddi_mfd && mddi_pipe) {
|
|
mdp4_mddi_dma_busy_wait(mddi_mfd);
|
|
mdp4_dma_s_update_lcd(mddi_mfd, mddi_pipe);
|
|
mdp4_mddi_dma_s_kickoff(mddi_mfd, mddi_pipe);
|
|
mddi_mfd->dma_update_flag = 1;
|
|
}
|
|
}
|
|
|
|
void mdp4_mddi_overlay(struct msm_fb_data_type *mfd)
|
|
{
|
|
mutex_lock(&mfd->dma->ov_mutex);
|
|
|
|
if (mfd && mfd->panel_power_on) {
|
|
mdp4_mddi_dma_busy_wait(mfd);
|
|
|
|
if (mddi_pipe && mddi_pipe->ov_blt_addr)
|
|
mdp4_mddi_blt_dmap_busy_wait(mfd);
|
|
mdp4_overlay_mdp_perf_upd(mfd, 0);
|
|
mdp4_overlay_update_lcd(mfd);
|
|
|
|
mdp4_overlay_mdp_perf_upd(mfd, 1);
|
|
if (mdp_hw_revision < MDP4_REVISION_V2_1) {
|
|
/* dmas dmap switch */
|
|
if (mdp4_overlay_mixer_play(mddi_pipe->mixer_num)
|
|
== 0) {
|
|
mdp4_dma_s_update_lcd(mfd, mddi_pipe);
|
|
mdp4_mddi_dma_s_kickoff(mfd, mddi_pipe);
|
|
} else
|
|
mdp4_mddi_kickoff_ui(mfd, mddi_pipe);
|
|
} else /* no dams dmap switch */
|
|
mdp4_mddi_kickoff_ui(mfd, mddi_pipe);
|
|
|
|
/* signal if pan function is waiting for the update completion */
|
|
if (mfd->pan_waiting) {
|
|
mfd->pan_waiting = FALSE;
|
|
complete(&mfd->pan_comp);
|
|
}
|
|
}
|
|
mutex_unlock(&mfd->dma->ov_mutex);
|
|
}
|
|
|
|
int mdp4_mddi_overlay_cursor(struct fb_info *info, struct fb_cursor *cursor)
|
|
{
|
|
struct msm_fb_data_type *mfd = info->par;
|
|
mutex_lock(&mfd->dma->ov_mutex);
|
|
if (mfd && mfd->panel_power_on) {
|
|
mdp4_mddi_dma_busy_wait(mfd);
|
|
mdp_hw_cursor_update(info, cursor);
|
|
}
|
|
mutex_unlock(&mfd->dma->ov_mutex);
|
|
return 0;
|
|
}
|