2 * linux/drivers/video/mmp/hw/mmp_ctrl.c
3 * Marvell MMP series Display Controller support
5 * Copyright (C) 2012 Marvell Technology Group Ltd.
6 * Authors: Guoqing Li <ligq@marvell.com>
7 * Lisa Du <cldu@marvell.com>
8 * Zhou Zhu <zzhu3@marvell.com>
10 * This program is free software; you can redistribute it and/or modify it
11 * under the terms of the GNU General Public License as published by the
12 * Free Software Foundation; either version 2 of the License, or (at your
13 * option) any later version.
15 * This program is distributed in the hope that it will be useful, but WITHOUT
16 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
17 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
20 * You should have received a copy of the GNU General Public License along with
21 * this program. If not, see <http://www.gnu.org/licenses/>.
24 #include <linux/module.h>
25 #include <linux/moduleparam.h>
26 #include <linux/kernel.h>
27 #include <linux/errno.h>
28 #include <linux/string.h>
29 #include <linux/interrupt.h>
30 #include <linux/slab.h>
31 #include <linux/delay.h>
32 #include <linux/platform_device.h>
33 #include <linux/dma-mapping.h>
34 #include <linux/clk.h>
35 #include <linux/err.h>
36 #include <linux/vmalloc.h>
37 #include <linux/uaccess.h>
38 #include <linux/kthread.h>
43 static irqreturn_t
ctrl_handle_irq(int irq
, void *dev_id
)
45 struct mmphw_ctrl
*ctrl
= (struct mmphw_ctrl
*)dev_id
;
48 isr
= readl_relaxed(ctrl
->reg_base
+ SPU_IRQ_ISR
);
49 imask
= readl_relaxed(ctrl
->reg_base
+ SPU_IRQ_ENA
);
52 /* clear clock only */
53 tmp
= readl_relaxed(ctrl
->reg_base
+ SPU_IRQ_ISR
);
55 writel_relaxed(~isr
, ctrl
->reg_base
+ SPU_IRQ_ISR
);
56 } while ((isr
= readl_relaxed(ctrl
->reg_base
+ SPU_IRQ_ISR
)) & imask
);
61 static u32
fmt_to_reg(struct mmp_overlay
*overlay
, int pix_fmt
)
63 u32 rbswap
= 0, uvswap
= 0, yuvswap
= 0,
65 vid
= overlay_is_vid(overlay
);
70 case PIXFMT_RGB888PACK
:
71 case PIXFMT_RGB888UNPACK
:
95 case PIXFMT_RGB888PACK
:
96 case PIXFMT_BGR888PACK
:
99 case PIXFMT_RGB888UNPACK
:
100 case PIXFMT_BGR888UNPACK
:
127 return (dma_palette(0) | dma_fmt(vid
, val
) |
128 dma_swaprb(vid
, rbswap
) | dma_swapuv(vid
, uvswap
) |
129 dma_swapyuv(vid
, yuvswap
) | dma_csc(vid
, csc_en
));
132 static void dmafetch_set_fmt(struct mmp_overlay
*overlay
)
135 struct mmp_path
*path
= overlay
->path
;
136 tmp
= readl_relaxed(ctrl_regs(path
) + dma_ctrl(0, path
->id
));
137 tmp
&= ~dma_mask(overlay_is_vid(overlay
));
138 tmp
|= fmt_to_reg(overlay
, overlay
->win
.pix_fmt
);
139 writel_relaxed(tmp
, ctrl_regs(path
) + dma_ctrl(0, path
->id
));
142 static void overlay_set_win(struct mmp_overlay
*overlay
, struct mmp_win
*win
)
144 struct lcd_regs
*regs
= path_regs(overlay
->path
);
146 /* assert win supported */
147 memcpy(&overlay
->win
, win
, sizeof(struct mmp_win
));
149 mutex_lock(&overlay
->access_ok
);
151 if (overlay_is_vid(overlay
)) {
152 writel_relaxed(win
->pitch
[0], ®s
->v_pitch_yc
);
153 writel_relaxed(win
->pitch
[2] << 16 |
154 win
->pitch
[1], ®s
->v_pitch_uv
);
156 writel_relaxed((win
->ysrc
<< 16) | win
->xsrc
, ®s
->v_size
);
157 writel_relaxed((win
->ydst
<< 16) | win
->xdst
, ®s
->v_size_z
);
158 writel_relaxed(win
->ypos
<< 16 | win
->xpos
, ®s
->v_start
);
160 writel_relaxed(win
->pitch
[0], ®s
->g_pitch
);
162 writel_relaxed((win
->ysrc
<< 16) | win
->xsrc
, ®s
->g_size
);
163 writel_relaxed((win
->ydst
<< 16) | win
->xdst
, ®s
->g_size_z
);
164 writel_relaxed(win
->ypos
<< 16 | win
->xpos
, ®s
->g_start
);
167 dmafetch_set_fmt(overlay
);
168 mutex_unlock(&overlay
->access_ok
);
171 static void dmafetch_onoff(struct mmp_overlay
*overlay
, int on
)
173 u32 mask
= overlay_is_vid(overlay
) ? CFG_DMA_ENA_MASK
:
175 u32 enable
= overlay_is_vid(overlay
) ? CFG_DMA_ENA(1) : CFG_GRA_ENA(1);
177 struct mmp_path
*path
= overlay
->path
;
179 mutex_lock(&overlay
->access_ok
);
180 tmp
= readl_relaxed(ctrl_regs(path
) + dma_ctrl(0, path
->id
));
182 tmp
|= (on
? enable
: 0);
183 writel(tmp
, ctrl_regs(path
) + dma_ctrl(0, path
->id
));
184 mutex_unlock(&overlay
->access_ok
);
187 static void path_enabledisable(struct mmp_path
*path
, int on
)
190 mutex_lock(&path
->access_ok
);
191 tmp
= readl_relaxed(ctrl_regs(path
) + LCD_SCLK(path
));
193 tmp
&= ~SCLK_DISABLE
;
196 writel_relaxed(tmp
, ctrl_regs(path
) + LCD_SCLK(path
));
197 mutex_unlock(&path
->access_ok
);
200 static void path_onoff(struct mmp_path
*path
, int on
)
202 if (path
->status
== on
) {
203 dev_info(path
->dev
, "path %s is already %s\n",
204 path
->name
, stat_name(path
->status
));
209 path_enabledisable(path
, 1);
211 if (path
->panel
&& path
->panel
->set_onoff
)
212 path
->panel
->set_onoff(path
->panel
, 1);
214 if (path
->panel
&& path
->panel
->set_onoff
)
215 path
->panel
->set_onoff(path
->panel
, 0);
217 path_enabledisable(path
, 0);
222 static void overlay_set_onoff(struct mmp_overlay
*overlay
, int on
)
224 if (overlay
->status
== on
) {
225 dev_info(overlay_to_ctrl(overlay
)->dev
, "overlay %s is already %s\n",
226 overlay
->path
->name
, stat_name(overlay
->status
));
229 overlay
->status
= on
;
230 dmafetch_onoff(overlay
, on
);
231 if (overlay
->path
->ops
.check_status(overlay
->path
)
232 != overlay
->path
->status
)
233 path_onoff(overlay
->path
, on
);
236 static void overlay_set_fetch(struct mmp_overlay
*overlay
, int fetch_id
)
238 overlay
->dmafetch_id
= fetch_id
;
241 static int overlay_set_addr(struct mmp_overlay
*overlay
, struct mmp_addr
*addr
)
243 struct lcd_regs
*regs
= path_regs(overlay
->path
);
245 /* FIXME: assert addr supported */
246 memcpy(&overlay
->addr
, addr
, sizeof(struct mmp_addr
));
248 if (overlay_is_vid(overlay
)) {
249 writel_relaxed(addr
->phys
[0], ®s
->v_y0
);
250 writel_relaxed(addr
->phys
[1], ®s
->v_u0
);
251 writel_relaxed(addr
->phys
[2], ®s
->v_v0
);
253 writel_relaxed(addr
->phys
[0], ®s
->g_0
);
255 return overlay
->addr
.phys
[0];
258 static void path_set_mode(struct mmp_path
*path
, struct mmp_mode
*mode
)
260 struct lcd_regs
*regs
= path_regs(path
);
261 u32 total_x
, total_y
, vsync_ctrl
, tmp
, sclk_src
, sclk_div
,
262 link_config
= path_to_path_plat(path
)->link_config
,
263 dsi_rbswap
= path_to_path_plat(path
)->link_config
;
265 /* FIXME: assert videomode supported */
266 memcpy(&path
->mode
, mode
, sizeof(struct mmp_mode
));
268 mutex_lock(&path
->access_ok
);
270 /* polarity of timing signals */
271 tmp
= readl_relaxed(ctrl_regs(path
) + intf_ctrl(path
->id
)) & 0x1;
272 tmp
|= mode
->vsync_invert
? 0 : 0x8;
273 tmp
|= mode
->hsync_invert
? 0 : 0x4;
274 tmp
|= link_config
& CFG_DUMBMODE_MASK
;
275 tmp
|= CFG_DUMB_ENA(1);
276 writel_relaxed(tmp
, ctrl_regs(path
) + intf_ctrl(path
->id
));
278 /* interface rb_swap setting */
279 tmp
= readl_relaxed(ctrl_regs(path
) + intf_rbswap_ctrl(path
->id
)) &
280 (~(CFG_INTFRBSWAP_MASK
));
281 tmp
|= dsi_rbswap
& CFG_INTFRBSWAP_MASK
;
282 writel_relaxed(tmp
, ctrl_regs(path
) + intf_rbswap_ctrl(path
->id
));
284 writel_relaxed((mode
->yres
<< 16) | mode
->xres
, ®s
->screen_active
);
285 writel_relaxed((mode
->left_margin
<< 16) | mode
->right_margin
,
286 ®s
->screen_h_porch
);
287 writel_relaxed((mode
->upper_margin
<< 16) | mode
->lower_margin
,
288 ®s
->screen_v_porch
);
289 total_x
= mode
->xres
+ mode
->left_margin
+ mode
->right_margin
+
291 total_y
= mode
->yres
+ mode
->upper_margin
+ mode
->lower_margin
+
293 writel_relaxed((total_y
<< 16) | total_x
, ®s
->screen_size
);
296 if (path
->output_type
== PATH_OUT_DSI
)
297 vsync_ctrl
= 0x01330133;
299 vsync_ctrl
= ((mode
->xres
+ mode
->right_margin
) << 16)
300 | (mode
->xres
+ mode
->right_margin
);
301 writel_relaxed(vsync_ctrl
, ®s
->vsync_ctrl
);
303 /* set pixclock div */
304 sclk_src
= clk_get_rate(path_to_ctrl(path
)->clk
);
305 sclk_div
= sclk_src
/ mode
->pixclock_freq
;
306 if (sclk_div
* mode
->pixclock_freq
< sclk_src
)
309 dev_info(path
->dev
, "%s sclk_src %d sclk_div 0x%x pclk %d\n",
310 __func__
, sclk_src
, sclk_div
, mode
->pixclock_freq
);
312 tmp
= readl_relaxed(ctrl_regs(path
) + LCD_SCLK(path
));
313 tmp
&= ~CLK_INT_DIV_MASK
;
315 writel_relaxed(tmp
, ctrl_regs(path
) + LCD_SCLK(path
));
317 mutex_unlock(&path
->access_ok
);
320 static struct mmp_overlay_ops mmphw_overlay_ops
= {
321 .set_fetch
= overlay_set_fetch
,
322 .set_onoff
= overlay_set_onoff
,
323 .set_win
= overlay_set_win
,
324 .set_addr
= overlay_set_addr
,
327 static void ctrl_set_default(struct mmphw_ctrl
*ctrl
)
332 * LCD Global control(LCD_TOP_CTRL) should be configed before
333 * any other LCD registers read/write, or there maybe issues.
335 tmp
= readl_relaxed(ctrl
->reg_base
+ LCD_TOP_CTRL
);
337 writel_relaxed(tmp
, ctrl
->reg_base
+ LCD_TOP_CTRL
);
340 /* disable all interrupts */
341 irq_mask
= path_imasks(0) | err_imask(0) |
342 path_imasks(1) | err_imask(1);
343 tmp
= readl_relaxed(ctrl
->reg_base
+ SPU_IRQ_ENA
);
346 writel_relaxed(tmp
, ctrl
->reg_base
+ SPU_IRQ_ENA
);
349 static void path_set_default(struct mmp_path
*path
)
351 struct lcd_regs
*regs
= path_regs(path
);
352 u32 dma_ctrl1
, mask
, tmp
, path_config
;
354 path_config
= path_to_path_plat(path
)->path_config
;
356 /* Configure IOPAD: should be parallel only */
357 if (PATH_OUT_PARALLEL
== path
->output_type
) {
358 mask
= CFG_IOPADMODE_MASK
| CFG_BURST_MASK
| CFG_BOUNDARY_MASK
;
359 tmp
= readl_relaxed(ctrl_regs(path
) + SPU_IOPAD_CONTROL
);
362 writel_relaxed(tmp
, ctrl_regs(path
) + SPU_IOPAD_CONTROL
);
365 /* Select path clock source */
366 tmp
= readl_relaxed(ctrl_regs(path
) + LCD_SCLK(path
));
367 tmp
&= ~SCLK_SRC_SEL_MASK
;
369 writel_relaxed(tmp
, ctrl_regs(path
) + LCD_SCLK(path
));
372 * Configure default bits: vsync triggers DMA,
373 * power save enable, configure alpha registers to
374 * display 100% graphics, and set pixel command.
376 dma_ctrl1
= 0x2032ff81;
378 dma_ctrl1
|= CFG_VSYNC_INV_MASK
;
379 writel_relaxed(dma_ctrl1
, ctrl_regs(path
) + dma_ctrl(1, path
->id
));
381 /* Configure default register values */
382 writel_relaxed(0x00000000, ®s
->blank_color
);
383 writel_relaxed(0x00000000, ®s
->g_1
);
384 writel_relaxed(0x00000000, ®s
->g_start
);
387 * 1.enable multiple burst request in DMA AXI
388 * bus arbiter for faster read if not tv path;
389 * 2.enable horizontal smooth filter;
391 mask
= CFG_GRA_HSMOOTH_MASK
| CFG_DMA_HSMOOTH_MASK
| CFG_ARBFAST_ENA(1);
392 tmp
= readl_relaxed(ctrl_regs(path
) + dma_ctrl(0, path
->id
));
394 if (PATH_TV
== path
->id
)
395 tmp
&= ~CFG_ARBFAST_ENA(1);
396 writel_relaxed(tmp
, ctrl_regs(path
) + dma_ctrl(0, path
->id
));
399 static int path_init(struct mmphw_path_plat
*path_plat
,
400 struct mmp_mach_path_config
*config
)
402 struct mmphw_ctrl
*ctrl
= path_plat
->ctrl
;
403 struct mmp_path_info
*path_info
;
404 struct mmp_path
*path
= NULL
;
406 dev_info(ctrl
->dev
, "%s: %s\n", __func__
, config
->name
);
408 /* init driver data */
409 path_info
= kzalloc(sizeof(*path_info
), GFP_KERNEL
);
413 path_info
->name
= config
->name
;
414 path_info
->id
= path_plat
->id
;
415 path_info
->dev
= ctrl
->dev
;
416 path_info
->overlay_num
= config
->overlay_num
;
417 path_info
->overlay_ops
= &mmphw_overlay_ops
;
418 path_info
->set_mode
= path_set_mode
;
419 path_info
->plat_data
= path_plat
;
421 /* create/register platform device */
422 path
= mmp_register_path(path_info
);
427 path_plat
->path
= path
;
428 path_plat
->path_config
= config
->path_config
;
429 path_plat
->link_config
= config
->link_config
;
430 path_plat
->dsi_rbswap
= config
->dsi_rbswap
;
431 path_set_default(path
);
437 static void path_deinit(struct mmphw_path_plat
*path_plat
)
442 mmp_unregister_path(path_plat
->path
);
445 static int mmphw_probe(struct platform_device
*pdev
)
447 struct mmp_mach_plat_info
*mi
;
448 struct resource
*res
;
449 int ret
, i
, size
, irq
;
450 struct mmphw_path_plat
*path_plat
;
451 struct mmphw_ctrl
*ctrl
= NULL
;
453 /* get resources from platform data */
454 res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
456 dev_err(&pdev
->dev
, "%s: no IO memory defined\n", __func__
);
461 irq
= platform_get_irq(pdev
, 0);
463 dev_err(&pdev
->dev
, "%s: no IRQ defined\n", __func__
);
468 /* get configs from platform data */
469 mi
= pdev
->dev
.platform_data
;
470 if (mi
== NULL
|| !mi
->path_num
|| !mi
->paths
) {
471 dev_err(&pdev
->dev
, "%s: no platform data defined\n", __func__
);
477 size
= sizeof(struct mmphw_ctrl
) + sizeof(struct mmphw_path_plat
) *
479 ctrl
= devm_kzalloc(&pdev
->dev
, size
, GFP_KERNEL
);
485 ctrl
->name
= mi
->name
;
486 ctrl
->path_num
= mi
->path_num
;
487 ctrl
->dev
= &pdev
->dev
;
489 platform_set_drvdata(pdev
, ctrl
);
490 mutex_init(&ctrl
->access_ok
);
493 if (!devm_request_mem_region(ctrl
->dev
, res
->start
,
494 resource_size(res
), ctrl
->name
)) {
496 "can't request region for resource %pR\n", res
);
501 ctrl
->reg_base
= devm_ioremap_nocache(ctrl
->dev
,
502 res
->start
, resource_size(res
));
503 if (ctrl
->reg_base
== NULL
) {
504 dev_err(ctrl
->dev
, "%s: res %pR map failed\n", __func__
, res
);
510 ret
= devm_request_irq(ctrl
->dev
, ctrl
->irq
, ctrl_handle_irq
,
511 IRQF_SHARED
, "lcd_controller", ctrl
);
513 dev_err(ctrl
->dev
, "%s unable to request IRQ %d\n",
514 __func__
, ctrl
->irq
);
520 ctrl
->clk
= devm_clk_get(ctrl
->dev
, mi
->clk_name
);
521 if (IS_ERR(ctrl
->clk
)) {
522 dev_err(ctrl
->dev
, "unable to get clk %s\n", mi
->clk_name
);
526 clk_prepare_enable(ctrl
->clk
);
528 /* init global regs */
529 ctrl_set_default(ctrl
);
531 /* init pathes from machine info and register them */
532 for (i
= 0; i
< ctrl
->path_num
; i
++) {
533 /* get from config and machine info */
534 path_plat
= &ctrl
->path_plats
[i
];
536 path_plat
->ctrl
= ctrl
;
539 if (!path_init(path_plat
, &mi
->paths
[i
])) {
541 goto failed_path_init
;
545 #ifdef CONFIG_MMP_DISP_SPI
546 ret
= lcd_spi_register(ctrl
);
548 goto failed_path_init
;
551 dev_info(ctrl
->dev
, "device init done\n");
556 for (i
= 0; i
< ctrl
->path_num
; i
++) {
557 path_plat
= &ctrl
->path_plats
[i
];
558 path_deinit(path_plat
);
561 clk_disable_unprepare(ctrl
->clk
);
563 dev_err(&pdev
->dev
, "device init failed\n");
568 static struct platform_driver mmphw_driver
= {
572 .probe
= mmphw_probe
,
575 static int mmphw_init(void)
577 return platform_driver_register(&mmphw_driver
);
579 module_init(mmphw_init
);
581 MODULE_AUTHOR("Li Guoqing<ligq@marvell.com>");
582 MODULE_DESCRIPTION("Framebuffer driver for mmp");
583 MODULE_LICENSE("GPL");