1 /* drivers/video/msm_fb/mdp.c
3 * MSM MDP Interface (used by framebuffer core)
5 * Copyright (C) 2007 QUALCOMM Incorporated
6 * Copyright (C) 2007 Google Incorporated
8 * This software is licensed under the terms of the GNU General Public
9 * License version 2, as published by the Free Software Foundation, and
10 * may be copied, distributed, and modified under those terms.
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
18 #include <linux/kernel.h>
20 #include <linux/msm_mdp.h>
21 #include <linux/interrupt.h>
22 #include <linux/wait.h>
23 #include <linux/clk.h>
24 #include <linux/file.h>
25 #ifdef CONFIG_ANDROID_PMEM
26 #include <linux/android_pmem.h>
28 #include <linux/major.h>
30 #include <mach/msm_iomap.h>
31 #include <mach/msm_fb.h>
32 #include <linux/platform_device.h>
36 struct class *mdp_class
;
38 #define MDP_CMD_DEBUG_ACCESS_BASE (0x10000)
40 static uint16_t mdp_default_ccs
[] = {
41 0x254, 0x000, 0x331, 0x254, 0xF38, 0xE61, 0x254, 0x409, 0x000,
45 static DECLARE_WAIT_QUEUE_HEAD(mdp_dma2_waitqueue
);
46 static DECLARE_WAIT_QUEUE_HEAD(mdp_ppp_waitqueue
);
47 static struct msmfb_callback
*dma_callback
;
48 static struct clk
*clk
;
49 static unsigned int mdp_irq_mask
;
50 static DEFINE_SPINLOCK(mdp_lock
);
51 DEFINE_MUTEX(mdp_mutex
);
53 static int enable_mdp_irq(struct mdp_info
*mdp
, uint32_t mask
)
55 unsigned long irq_flags
;
60 spin_lock_irqsave(&mdp_lock
, irq_flags
);
61 /* if the mask bits are already set return an error, this interrupt
62 * is already enabled */
63 if (mdp_irq_mask
& mask
) {
64 printk(KERN_ERR
"mdp irq already on already on %x %x\n",
68 /* if the mdp irq is not already enabled enable it */
75 /* update the irq mask to reflect the fact that the interrupt is
78 spin_unlock_irqrestore(&mdp_lock
, irq_flags
);
82 static int locked_disable_mdp_irq(struct mdp_info
*mdp
, uint32_t mask
)
84 /* this interrupt is already disabled! */
85 if (!(mdp_irq_mask
& mask
)) {
86 printk(KERN_ERR
"mdp irq already off %x %x\n",
90 /* update the irq mask to reflect the fact that the interrupt is
92 mdp_irq_mask
&= ~(mask
);
93 /* if no one is waiting on the interrupt, disable it */
95 disable_irq(mdp
->irq
);
102 static int disable_mdp_irq(struct mdp_info
*mdp
, uint32_t mask
)
104 unsigned long irq_flags
;
107 spin_lock_irqsave(&mdp_lock
, irq_flags
);
108 ret
= locked_disable_mdp_irq(mdp
, mask
);
109 spin_unlock_irqrestore(&mdp_lock
, irq_flags
);
113 static irqreturn_t
mdp_isr(int irq
, void *data
)
116 unsigned long irq_flags
;
117 struct mdp_info
*mdp
= data
;
119 spin_lock_irqsave(&mdp_lock
, irq_flags
);
121 status
= mdp_readl(mdp
, MDP_INTR_STATUS
);
122 mdp_writel(mdp
, status
, MDP_INTR_CLEAR
);
124 status
&= mdp_irq_mask
;
125 if (status
& DL0_DMA2_TERM_DONE
) {
127 dma_callback
->func(dma_callback
);
130 wake_up(&mdp_dma2_waitqueue
);
133 if (status
& DL0_ROI_DONE
)
134 wake_up(&mdp_ppp_waitqueue
);
137 locked_disable_mdp_irq(mdp
, status
);
139 spin_unlock_irqrestore(&mdp_lock
, irq_flags
);
143 static uint32_t mdp_check_mask(uint32_t mask
)
146 unsigned long irq_flags
;
148 spin_lock_irqsave(&mdp_lock
, irq_flags
);
149 ret
= mdp_irq_mask
& mask
;
150 spin_unlock_irqrestore(&mdp_lock
, irq_flags
);
154 static int mdp_wait(struct mdp_info
*mdp
, uint32_t mask
, wait_queue_head_t
*wq
)
157 unsigned long irq_flags
;
159 wait_event_timeout(*wq
, !mdp_check_mask(mask
), HZ
);
161 spin_lock_irqsave(&mdp_lock
, irq_flags
);
162 if (mdp_irq_mask
& mask
) {
163 locked_disable_mdp_irq(mdp
, mask
);
164 printk(KERN_WARNING
"timeout waiting for mdp to complete %x\n",
168 spin_unlock_irqrestore(&mdp_lock
, irq_flags
);
173 void mdp_dma_wait(struct mdp_device
*mdp_dev
)
175 #define MDP_MAX_TIMEOUTS 20
176 static int timeout_count
;
177 struct mdp_info
*mdp
= container_of(mdp_dev
, struct mdp_info
, mdp_dev
);
179 if (mdp_wait(mdp
, DL0_DMA2_TERM_DONE
, &mdp_dma2_waitqueue
) == -ETIMEDOUT
)
184 if (timeout_count
> MDP_MAX_TIMEOUTS
) {
185 printk(KERN_ERR
"mdp: dma failed %d times, somethings wrong!\n",
191 static int mdp_ppp_wait(struct mdp_info
*mdp
)
193 return mdp_wait(mdp
, DL0_ROI_DONE
, &mdp_ppp_waitqueue
);
196 void mdp_dma_to_mddi(struct mdp_info
*mdp
, uint32_t addr
, uint32_t stride
,
197 uint32_t width
, uint32_t height
, uint32_t x
, uint32_t y
,
198 struct msmfb_callback
*callback
)
201 uint16_t ld_param
= 0; /* 0=PRIM, 1=SECD, 2=EXT */
203 if (enable_mdp_irq(mdp
, DL0_DMA2_TERM_DONE
)) {
204 printk(KERN_ERR
"mdp_dma_to_mddi: busy\n");
208 dma_callback
= callback
;
210 dma2_cfg
= DMA_PACK_TIGHT
|
212 DMA_PACK_PATTERN_RGB
|
214 DMA_IBUF_NONCONTIGUOUS
;
216 dma2_cfg
|= DMA_IBUF_FORMAT_RGB565
;
218 dma2_cfg
|= DMA_OUT_SEL_MDDI
;
220 dma2_cfg
|= DMA_MDDI_DMAOUT_LCD_SEL_PRIMARY
;
222 dma2_cfg
|= DMA_DITHER_EN
;
224 /* setup size, address, and stride */
225 mdp_writel(mdp
, (height
<< 16) | (width
),
226 MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0184);
227 mdp_writel(mdp
, addr
, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0188);
228 mdp_writel(mdp
, stride
, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x018C);
231 dma2_cfg
|= DMA_DSTC0G_6BITS
| DMA_DSTC1B_6BITS
| DMA_DSTC2R_6BITS
;
233 /* set y & x offset and MDDI transaction parameters */
234 mdp_writel(mdp
, (y
<< 16) | (x
), MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0194);
235 mdp_writel(mdp
, ld_param
, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01a0);
236 mdp_writel(mdp
, (MDDI_VDO_PACKET_DESC
<< 16) | MDDI_VDO_PACKET_PRIM
,
237 MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01a4);
239 mdp_writel(mdp
, dma2_cfg
, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0180);
242 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0044);
245 void mdp_dma(struct mdp_device
*mdp_dev
, uint32_t addr
, uint32_t stride
,
246 uint32_t width
, uint32_t height
, uint32_t x
, uint32_t y
,
247 struct msmfb_callback
*callback
, int interface
)
249 struct mdp_info
*mdp
= container_of(mdp_dev
, struct mdp_info
, mdp_dev
);
251 if (interface
== MSM_MDDI_PMDH_INTERFACE
) {
252 mdp_dma_to_mddi(mdp
, addr
, stride
, width
, height
, x
, y
,
257 int get_img(struct mdp_img
*img
, struct fb_info
*info
,
258 unsigned long *start
, unsigned long *len
,
261 int put_needed
, ret
= 0;
263 unsigned long vstart
;
265 #ifdef CONFIG_ANDROID_PMEM
266 if (!get_pmem_file(img
->memory_id
, start
, &vstart
, len
, filep
))
270 file
= fget_light(img
->memory_id
, &put_needed
);
274 if (MAJOR(file
->f_dentry
->d_inode
->i_rdev
) == FB_MAJOR
) {
275 *start
= info
->fix
.smem_start
;
276 *len
= info
->fix
.smem_len
;
279 fput_light(file
, put_needed
);
284 void put_img(struct file
*src_file
, struct file
*dst_file
)
286 #ifdef CONFIG_ANDROID_PMEM
288 put_pmem_file(src_file
);
290 put_pmem_file(dst_file
);
294 int mdp_blit(struct mdp_device
*mdp_dev
, struct fb_info
*fb
,
295 struct mdp_blit_req
*req
)
298 unsigned long src_start
= 0, src_len
= 0, dst_start
= 0, dst_len
= 0;
299 struct mdp_info
*mdp
= container_of(mdp_dev
, struct mdp_info
, mdp_dev
);
300 struct file
*src_file
= 0, *dst_file
= 0;
302 /* WORKAROUND FOR HARDWARE BUG IN BG TILE FETCH */
303 if (unlikely(req
->src_rect
.h
== 0 ||
304 req
->src_rect
.w
== 0)) {
305 printk(KERN_ERR
"mpd_ppp: src img of zero size!\n");
308 if (unlikely(req
->dst_rect
.h
== 0 ||
309 req
->dst_rect
.w
== 0))
312 /* do this first so that if this fails, the caller can always
313 * safely call put_img */
314 if (unlikely(get_img(&req
->src
, fb
, &src_start
, &src_len
, &src_file
))) {
315 printk(KERN_ERR
"mpd_ppp: could not retrieve src image from "
320 if (unlikely(get_img(&req
->dst
, fb
, &dst_start
, &dst_len
, &dst_file
))) {
321 printk(KERN_ERR
"mpd_ppp: could not retrieve dst image from "
323 #ifdef CONFIG_ANDROID_PMEM
324 put_pmem_file(src_file
);
328 mutex_lock(&mdp_mutex
);
330 /* transp_masking unimplemented */
331 req
->transp_mask
= MDP_TRANSP_NOP
;
332 if (unlikely((req
->transp_mask
!= MDP_TRANSP_NOP
||
333 req
->alpha
!= MDP_ALPHA_NOP
||
334 HAS_ALPHA(req
->src
.format
)) &&
335 (req
->flags
& MDP_ROT_90
&&
336 req
->dst_rect
.w
<= 16 && req
->dst_rect
.h
>= 16))) {
338 unsigned int tiles
= req
->dst_rect
.h
/ 16;
339 unsigned int remainder
= req
->dst_rect
.h
% 16;
340 req
->src_rect
.w
= 16*req
->src_rect
.w
/ req
->dst_rect
.h
;
341 req
->dst_rect
.h
= 16;
342 for (i
= 0; i
< tiles
; i
++) {
343 enable_mdp_irq(mdp
, DL0_ROI_DONE
);
344 ret
= mdp_ppp_blit(mdp
, req
, src_file
, src_start
,
345 src_len
, dst_file
, dst_start
,
349 ret
= mdp_ppp_wait(mdp
);
351 goto err_wait_failed
;
352 req
->dst_rect
.y
+= 16;
353 req
->src_rect
.x
+= req
->src_rect
.w
;
357 req
->src_rect
.w
= remainder
*req
->src_rect
.w
/ req
->dst_rect
.h
;
358 req
->dst_rect
.h
= remainder
;
360 enable_mdp_irq(mdp
, DL0_ROI_DONE
);
361 ret
= mdp_ppp_blit(mdp
, req
, src_file
, src_start
, src_len
, dst_file
,
366 ret
= mdp_ppp_wait(mdp
);
368 goto err_wait_failed
;
370 put_img(src_file
, dst_file
);
371 mutex_unlock(&mdp_mutex
);
374 disable_mdp_irq(mdp
, DL0_ROI_DONE
);
376 put_img(src_file
, dst_file
);
377 mutex_unlock(&mdp_mutex
);
381 void mdp_set_grp_disp(struct mdp_device
*mdp_dev
, unsigned disp_id
)
383 struct mdp_info
*mdp
= container_of(mdp_dev
, struct mdp_info
, mdp_dev
);
386 mdp_writel(mdp
, disp_id
, MDP_FULL_BYPASS_WORD43
);
389 int register_mdp_client(struct class_interface
*cint
)
392 pr_err("mdp: no mdp_class when registering mdp client\n");
395 cint
->class = mdp_class
;
396 return class_interface_register(cint
);
399 #include "mdp_csc_table.h"
400 #include "mdp_scale_tables.h"
402 int mdp_probe(struct platform_device
*pdev
)
404 struct resource
*resource
;
407 struct mdp_info
*mdp
;
409 resource
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
411 pr_err("mdp: can not get mdp mem resource!\n");
415 mdp
= kzalloc(sizeof(struct mdp_info
), GFP_KERNEL
);
419 mdp
->irq
= platform_get_irq(pdev
, 0);
421 pr_err("mdp: can not get mdp irq\n");
426 mdp
->base
= ioremap(resource
->start
,
427 resource
->end
- resource
->start
);
428 if (mdp
->base
== 0) {
429 printk(KERN_ERR
"msmfb: cannot allocate mdp regs!\n");
434 mdp
->mdp_dev
.dma
= mdp_dma
;
435 mdp
->mdp_dev
.dma_wait
= mdp_dma_wait
;
436 mdp
->mdp_dev
.blit
= mdp_blit
;
437 mdp
->mdp_dev
.set_grp_disp
= mdp_set_grp_disp
;
439 clk
= clk_get(&pdev
->dev
, "mdp_clk");
441 printk(KERN_INFO
"mdp: failed to get mdp clk");
445 ret
= request_irq(mdp
->irq
, mdp_isr
, IRQF_DISABLED
, "msm_mdp", mdp
);
447 goto error_request_irq
;
448 disable_irq(mdp
->irq
);
451 /* debug interface write access */
452 mdp_writel(mdp
, 1, 0x60);
454 mdp_writel(mdp
, MDP_ANY_INTR_MASK
, MDP_INTR_ENABLE
);
455 mdp_writel(mdp
, 1, MDP_EBI2_PORTMAP_MODE
);
457 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01f8);
458 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01fc);
460 for (n
= 0; n
< ARRAY_SIZE(csc_table
); n
++)
461 mdp_writel(mdp
, csc_table
[n
].val
, csc_table
[n
].reg
);
463 /* clear up unused fg/main registers */
464 /* comp.plane 2&3 ystride */
465 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0120);
467 /* unpacked pattern */
468 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x012c);
469 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0130);
470 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0134);
471 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0158);
472 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x015c);
473 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0160);
474 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0170);
475 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0174);
476 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x017c);
478 /* comp.plane 2 & 3 */
479 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0114);
480 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x0118);
482 /* clear unused bg registers */
483 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01c8);
484 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01d0);
485 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01dc);
486 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01e0);
487 mdp_writel(mdp
, 0, MDP_CMD_DEBUG_ACCESS_BASE
+ 0x01e4);
489 for (n
= 0; n
< ARRAY_SIZE(mdp_upscale_table
); n
++)
490 mdp_writel(mdp
, mdp_upscale_table
[n
].val
,
491 mdp_upscale_table
[n
].reg
);
493 for (n
= 0; n
< 9; n
++)
494 mdp_writel(mdp
, mdp_default_ccs
[n
], 0x40440 + 4 * n
);
495 mdp_writel(mdp
, mdp_default_ccs
[9], 0x40500 + 4 * 0);
496 mdp_writel(mdp
, mdp_default_ccs
[10], 0x40500 + 4 * 0);
497 mdp_writel(mdp
, mdp_default_ccs
[11], 0x40500 + 4 * 0);
499 /* register mdp device */
500 mdp
->mdp_dev
.dev
.parent
= &pdev
->dev
;
501 mdp
->mdp_dev
.dev
.class = mdp_class
;
502 snprintf(mdp
->mdp_dev
.dev
.bus_id
, BUS_ID_SIZE
, "mdp%d", pdev
->id
);
504 /* if you can remove the platform device you'd have to implement
506 mdp_dev.release = mdp_class; */
508 ret
= device_register(&mdp
->mdp_dev
.dev
);
510 goto error_device_register
;
513 error_device_register
:
514 free_irq(mdp
->irq
, mdp
);
523 static struct platform_driver msm_mdp_driver
= {
525 .driver
= {.name
= "msm_mdp"},
528 static int __init
mdp_init(void)
530 mdp_class
= class_create(THIS_MODULE
, "msm_mdp");
531 if (IS_ERR(mdp_class
)) {
532 printk(KERN_ERR
"Error creating mdp class\n");
533 return PTR_ERR(mdp_class
);
535 return platform_driver_register(&msm_mdp_driver
);
538 subsys_initcall(mdp_init
);