2 * Copyright (C) 2012 Russell King
3 * Rewritten from the dovefb driver, and Armada510 manuals.
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
10 #include <linux/component.h>
11 #include <linux/of_device.h>
12 #include <linux/platform_device.h>
14 #include <drm/drm_crtc_helper.h>
15 #include <drm/drm_plane_helper.h>
16 #include <drm/drm_atomic_helper.h>
17 #include "armada_crtc.h"
18 #include "armada_drm.h"
19 #include "armada_fb.h"
20 #include "armada_gem.h"
21 #include "armada_hw.h"
22 #include "armada_trace.h"
32 static const uint32_t armada_primary_formats
[] = {
50 * A note about interlacing. Let's consider HDMI 1920x1080i.
51 * The timing parameters we have from X are:
52 * Hact HsyA HsyI Htot Vact VsyA VsyI Vtot
53 * 1920 2448 2492 2640 1080 1084 1094 1125
54 * Which get translated to:
55 * Hact HsyA HsyI Htot Vact VsyA VsyI Vtot
56 * 1920 2448 2492 2640 540 542 547 562
58 * This is how it is defined by CEA-861-D - line and pixel numbers are
59 * referenced to the rising edge of VSYNC and HSYNC. Total clocks per
60 * line: 2640. The odd frame, the first active line is at line 21, and
61 * the even frame, the first active line is 584.
63 * LN: 560 561 562 563 567 568 569
64 * DE: ~~~|____________________________//__________________________
65 * HSYNC: ____|~|_____|~|_____|~|_____|~|_//__|~|_____|~|_____|~|_____
66 * VSYNC: _________________________|~~~~~~//~~~~~~~~~~~~~~~|__________
67 * 22 blanking lines. VSYNC at 1320 (referenced to the HSYNC rising edge).
69 * LN: 1123 1124 1125 1 5 6 7
70 * DE: ~~~|____________________________//__________________________
71 * HSYNC: ____|~|_____|~|_____|~|_____|~|_//__|~|_____|~|_____|~|_____
72 * VSYNC: ____________________|~~~~~~~~~~~//~~~~~~~~~~|_______________
75 * The Armada LCD Controller line and pixel numbers are, like X timings,
76 * referenced to the top left of the active frame.
78 * So, translating these to our LCD controller:
79 * Odd frame, 563 total lines, VSYNC at line 543-548, pixel 1128.
80 * Even frame, 562 total lines, VSYNC at line 542-547, pixel 2448.
81 * Note: Vsync front porch remains constant!
84 * vtotal = mode->crtc_vtotal + 1;
85 * vbackporch = mode->crtc_vsync_start - mode->crtc_vdisplay + 1;
86 * vhorizpos = mode->crtc_hsync_start - mode->crtc_htotal / 2
88 * vtotal = mode->crtc_vtotal;
89 * vbackporch = mode->crtc_vsync_start - mode->crtc_vdisplay;
90 * vhorizpos = mode->crtc_hsync_start;
92 * vfrontporch = mode->crtc_vtotal - mode->crtc_vsync_end;
94 * So, we need to reprogram these registers on each vsync event:
95 * LCD_SPU_V_PORCH, LCD_SPU_ADV_REG, LCD_SPUT_V_H_TOTAL
97 * Note: we do not use the frame done interrupts because these appear
98 * to happen too early, and lead to jitter on the display (presumably
99 * they occur at the end of the last active line, before the vsync back
100 * porch, which we're reprogramming.)
104 armada_drm_crtc_update_regs(struct armada_crtc
*dcrtc
, struct armada_regs
*regs
)
106 while (regs
->offset
!= ~0) {
107 void __iomem
*reg
= dcrtc
->base
+ regs
->offset
;
112 val
&= readl_relaxed(reg
);
113 writel_relaxed(val
| regs
->val
, reg
);
118 #define dpms_blanked(dpms) ((dpms) != DRM_MODE_DPMS_ON)
120 static void armada_drm_crtc_update(struct armada_crtc
*dcrtc
)
124 dumb_ctrl
= dcrtc
->cfg_dumb_ctrl
;
126 if (!dpms_blanked(dcrtc
->dpms
))
127 dumb_ctrl
|= CFG_DUMB_ENA
;
130 * When the dumb interface isn't in DUMB24_RGB888_0 mode, it might
131 * be using SPI or GPIO. If we set this to DUMB_BLANK, we will
132 * force LCD_D[23:0] to output blank color, overriding the GPIO or
133 * SPI usage. So leave it as-is unless in DUMB24_RGB888_0 mode.
135 if (dpms_blanked(dcrtc
->dpms
) &&
136 (dumb_ctrl
& DUMB_MASK
) == DUMB24_RGB888_0
) {
137 dumb_ctrl
&= ~DUMB_MASK
;
138 dumb_ctrl
|= DUMB_BLANK
;
142 * The documentation doesn't indicate what the normal state of
143 * the sync signals are. Sebastian Hesselbart kindly probed
144 * these signals on his board to determine their state.
146 * The non-inverted state of the sync signals is active high.
147 * Setting these bits makes the appropriate signal active low.
149 if (dcrtc
->crtc
.mode
.flags
& DRM_MODE_FLAG_NCSYNC
)
150 dumb_ctrl
|= CFG_INV_CSYNC
;
151 if (dcrtc
->crtc
.mode
.flags
& DRM_MODE_FLAG_NHSYNC
)
152 dumb_ctrl
|= CFG_INV_HSYNC
;
153 if (dcrtc
->crtc
.mode
.flags
& DRM_MODE_FLAG_NVSYNC
)
154 dumb_ctrl
|= CFG_INV_VSYNC
;
156 if (dcrtc
->dumb_ctrl
!= dumb_ctrl
) {
157 dcrtc
->dumb_ctrl
= dumb_ctrl
;
158 writel_relaxed(dumb_ctrl
, dcrtc
->base
+ LCD_SPU_DUMB_CTRL
);
162 void armada_drm_plane_calc_addrs(u32
*addrs
, struct drm_framebuffer
*fb
,
165 const struct drm_format_info
*format
= fb
->format
;
166 unsigned int num_planes
= format
->num_planes
;
167 u32 addr
= drm_fb_obj(fb
)->dev_addr
;
173 addrs
[0] = addr
+ fb
->offsets
[0] + y
* fb
->pitches
[0] +
179 for (i
= 1; i
< num_planes
; i
++)
180 addrs
[i
] = addr
+ fb
->offsets
[i
] + y
* fb
->pitches
[i
] +
186 static unsigned armada_drm_crtc_calc_fb(struct drm_framebuffer
*fb
,
187 int x
, int y
, struct armada_regs
*regs
, bool interlaced
)
189 unsigned pitch
= fb
->pitches
[0];
190 u32 addrs
[3], addr_odd
, addr_even
;
193 DRM_DEBUG_DRIVER("pitch %u x %d y %d bpp %d\n",
194 pitch
, x
, y
, fb
->format
->cpp
[0] * 8);
196 armada_drm_plane_calc_addrs(addrs
, fb
, x
, y
);
198 addr_odd
= addr_even
= addrs
[0];
205 /* write offset, base, and pitch */
206 armada_reg_queue_set(regs
, i
, addr_odd
, LCD_CFG_GRA_START_ADDR0
);
207 armada_reg_queue_set(regs
, i
, addr_even
, LCD_CFG_GRA_START_ADDR1
);
208 armada_reg_queue_mod(regs
, i
, pitch
, 0xffff, LCD_CFG_GRA_PITCH
);
213 static void armada_drm_plane_work_call(struct armada_crtc
*dcrtc
,
214 struct armada_plane_work
*work
,
215 void (*fn
)(struct armada_crtc
*, struct armada_plane_work
*))
217 struct armada_plane
*dplane
= drm_to_armada_plane(work
->plane
);
218 struct drm_pending_vblank_event
*event
;
219 struct drm_framebuffer
*fb
;
223 drm_crtc_vblank_put(&dcrtc
->crtc
);
228 struct drm_device
*dev
= dcrtc
->crtc
.dev
;
231 spin_lock_irqsave(&dev
->event_lock
, flags
);
233 drm_crtc_send_vblank_event(&dcrtc
->crtc
, event
);
235 __armada_drm_queue_unref_work(dev
, fb
);
236 spin_unlock_irqrestore(&dev
->event_lock
, flags
);
239 if (work
->need_kfree
)
242 wake_up(&dplane
->frame_wait
);
245 static void armada_drm_plane_work_run(struct armada_crtc
*dcrtc
,
246 struct drm_plane
*plane
)
248 struct armada_plane
*dplane
= drm_to_armada_plane(plane
);
249 struct armada_plane_work
*work
= xchg(&dplane
->work
, NULL
);
251 /* Handle any pending frame work. */
253 armada_drm_plane_work_call(dcrtc
, work
, work
->fn
);
256 int armada_drm_plane_work_queue(struct armada_crtc
*dcrtc
,
257 struct armada_plane_work
*work
)
259 struct armada_plane
*plane
= drm_to_armada_plane(work
->plane
);
262 ret
= drm_crtc_vblank_get(&dcrtc
->crtc
);
266 ret
= cmpxchg(&plane
->work
, NULL
, work
) ? -EBUSY
: 0;
268 drm_crtc_vblank_put(&dcrtc
->crtc
);
273 int armada_drm_plane_work_wait(struct armada_plane
*plane
, long timeout
)
275 return wait_event_timeout(plane
->frame_wait
, !plane
->work
, timeout
);
278 void armada_drm_plane_work_cancel(struct armada_crtc
*dcrtc
,
279 struct armada_plane
*dplane
)
281 struct armada_plane_work
*work
= xchg(&dplane
->work
, NULL
);
284 armada_drm_plane_work_call(dcrtc
, work
, work
->cancel
);
287 static void armada_drm_crtc_complete_frame_work(struct armada_crtc
*dcrtc
,
288 struct armada_plane_work
*work
)
292 spin_lock_irqsave(&dcrtc
->irq_lock
, flags
);
293 armada_drm_crtc_update_regs(dcrtc
, work
->regs
);
294 spin_unlock_irqrestore(&dcrtc
->irq_lock
, flags
);
297 static void armada_drm_crtc_complete_disable_work(struct armada_crtc
*dcrtc
,
298 struct armada_plane_work
*work
)
302 if (dcrtc
->plane
== work
->plane
)
305 spin_lock_irqsave(&dcrtc
->irq_lock
, flags
);
306 armada_drm_crtc_update_regs(dcrtc
, work
->regs
);
307 spin_unlock_irqrestore(&dcrtc
->irq_lock
, flags
);
310 static struct armada_plane_work
*
311 armada_drm_crtc_alloc_plane_work(struct drm_plane
*plane
)
313 struct armada_plane_work
*work
;
316 work
= kzalloc(sizeof(*work
), GFP_KERNEL
);
321 work
->fn
= armada_drm_crtc_complete_frame_work
;
322 work
->need_kfree
= true;
323 armada_reg_queue_end(work
->regs
, i
);
328 static void armada_drm_crtc_finish_fb(struct armada_crtc
*dcrtc
,
329 struct drm_framebuffer
*fb
, bool force
)
331 struct armada_plane_work
*work
;
337 /* Display is disabled, so just drop the old fb */
338 drm_framebuffer_put(fb
);
342 work
= armada_drm_crtc_alloc_plane_work(dcrtc
->crtc
.primary
);
346 if (armada_drm_plane_work_queue(dcrtc
, work
) == 0)
353 * Oops - just drop the reference immediately and hope for
354 * the best. The worst that will happen is the buffer gets
355 * reused before it has finished being displayed.
357 drm_framebuffer_put(fb
);
360 static void armada_drm_vblank_off(struct armada_crtc
*dcrtc
)
363 * Tell the DRM core that vblank IRQs aren't going to happen for
364 * a while. This cleans up any pending vblank events for us.
366 drm_crtc_vblank_off(&dcrtc
->crtc
);
367 armada_drm_plane_work_run(dcrtc
, dcrtc
->crtc
.primary
);
370 /* The mode_config.mutex will be held for this call */
371 static void armada_drm_crtc_dpms(struct drm_crtc
*crtc
, int dpms
)
373 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
375 if (dpms_blanked(dcrtc
->dpms
) != dpms_blanked(dpms
)) {
376 if (dpms_blanked(dpms
))
377 armada_drm_vblank_off(dcrtc
);
378 else if (!IS_ERR(dcrtc
->clk
))
379 WARN_ON(clk_prepare_enable(dcrtc
->clk
));
381 armada_drm_crtc_update(dcrtc
);
382 if (!dpms_blanked(dpms
))
383 drm_crtc_vblank_on(&dcrtc
->crtc
);
384 else if (!IS_ERR(dcrtc
->clk
))
385 clk_disable_unprepare(dcrtc
->clk
);
386 } else if (dcrtc
->dpms
!= dpms
) {
392 * Prepare for a mode set. Turn off overlay to ensure that we don't end
393 * up with the overlay size being bigger than the active screen size.
394 * We rely upon X refreshing this state after the mode set has completed.
396 * The mode_config.mutex will be held for this call
398 static void armada_drm_crtc_prepare(struct drm_crtc
*crtc
)
400 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
401 struct drm_plane
*plane
;
404 * If we have an overlay plane associated with this CRTC, disable
405 * it before the modeset to avoid its coordinates being outside
406 * the new mode parameters.
408 plane
= dcrtc
->plane
;
410 drm_plane_force_disable(plane
);
411 WARN_ON(!armada_drm_plane_work_wait(drm_to_armada_plane(plane
),
416 /* The mode_config.mutex will be held for this call */
417 static void armada_drm_crtc_commit(struct drm_crtc
*crtc
)
419 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
421 if (dcrtc
->dpms
!= DRM_MODE_DPMS_ON
) {
422 dcrtc
->dpms
= DRM_MODE_DPMS_ON
;
423 armada_drm_crtc_update(dcrtc
);
427 /* The mode_config.mutex will be held for this call */
428 static bool armada_drm_crtc_mode_fixup(struct drm_crtc
*crtc
,
429 const struct drm_display_mode
*mode
, struct drm_display_mode
*adj
)
431 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
434 /* We can't do interlaced modes if we don't have the SPU_ADV_REG */
435 if (!dcrtc
->variant
->has_spu_adv_reg
&&
436 adj
->flags
& DRM_MODE_FLAG_INTERLACE
)
439 /* Check whether the display mode is possible */
440 ret
= dcrtc
->variant
->compute_clock(dcrtc
, adj
, NULL
);
447 /* These are locked by dev->vbl_lock */
448 static void armada_drm_crtc_disable_irq(struct armada_crtc
*dcrtc
, u32 mask
)
450 if (dcrtc
->irq_ena
& mask
) {
451 dcrtc
->irq_ena
&= ~mask
;
452 writel(dcrtc
->irq_ena
, dcrtc
->base
+ LCD_SPU_IRQ_ENA
);
456 static void armada_drm_crtc_enable_irq(struct armada_crtc
*dcrtc
, u32 mask
)
458 if ((dcrtc
->irq_ena
& mask
) != mask
) {
459 dcrtc
->irq_ena
|= mask
;
460 writel(dcrtc
->irq_ena
, dcrtc
->base
+ LCD_SPU_IRQ_ENA
);
461 if (readl_relaxed(dcrtc
->base
+ LCD_SPU_IRQ_ISR
) & mask
)
462 writel(0, dcrtc
->base
+ LCD_SPU_IRQ_ISR
);
466 static void armada_drm_crtc_irq(struct armada_crtc
*dcrtc
, u32 stat
)
468 void __iomem
*base
= dcrtc
->base
;
469 struct drm_plane
*ovl_plane
;
471 if (stat
& DMA_FF_UNDERFLOW
)
472 DRM_ERROR("video underflow on crtc %u\n", dcrtc
->num
);
473 if (stat
& GRA_FF_UNDERFLOW
)
474 DRM_ERROR("graphics underflow on crtc %u\n", dcrtc
->num
);
476 if (stat
& VSYNC_IRQ
)
477 drm_crtc_handle_vblank(&dcrtc
->crtc
);
479 ovl_plane
= dcrtc
->plane
;
481 armada_drm_plane_work_run(dcrtc
, ovl_plane
);
483 spin_lock(&dcrtc
->irq_lock
);
484 if (stat
& GRA_FRAME_IRQ
&& dcrtc
->interlaced
) {
485 int i
= stat
& GRA_FRAME_IRQ0
? 0 : 1;
488 writel_relaxed(dcrtc
->v
[i
].spu_v_porch
, base
+ LCD_SPU_V_PORCH
);
489 writel_relaxed(dcrtc
->v
[i
].spu_v_h_total
,
490 base
+ LCD_SPUT_V_H_TOTAL
);
492 val
= readl_relaxed(base
+ LCD_SPU_ADV_REG
);
493 val
&= ~(ADV_VSYNC_L_OFF
| ADV_VSYNC_H_OFF
| ADV_VSYNCOFFEN
);
494 val
|= dcrtc
->v
[i
].spu_adv_reg
;
495 writel_relaxed(val
, base
+ LCD_SPU_ADV_REG
);
498 if (stat
& DUMB_FRAMEDONE
&& dcrtc
->cursor_update
) {
499 writel_relaxed(dcrtc
->cursor_hw_pos
,
500 base
+ LCD_SPU_HWC_OVSA_HPXL_VLN
);
501 writel_relaxed(dcrtc
->cursor_hw_sz
,
502 base
+ LCD_SPU_HWC_HPXL_VLN
);
503 armada_updatel(CFG_HWC_ENA
,
504 CFG_HWC_ENA
| CFG_HWC_1BITMOD
| CFG_HWC_1BITENA
,
505 base
+ LCD_SPU_DMA_CTRL0
);
506 dcrtc
->cursor_update
= false;
507 armada_drm_crtc_disable_irq(dcrtc
, DUMB_FRAMEDONE_ENA
);
510 spin_unlock(&dcrtc
->irq_lock
);
512 if (stat
& GRA_FRAME_IRQ
)
513 armada_drm_plane_work_run(dcrtc
, dcrtc
->crtc
.primary
);
516 static irqreturn_t
armada_drm_irq(int irq
, void *arg
)
518 struct armada_crtc
*dcrtc
= arg
;
519 u32 v
, stat
= readl_relaxed(dcrtc
->base
+ LCD_SPU_IRQ_ISR
);
522 * This is rediculous - rather than writing bits to clear, we
523 * have to set the actual status register value. This is racy.
525 writel_relaxed(0, dcrtc
->base
+ LCD_SPU_IRQ_ISR
);
527 trace_armada_drm_irq(&dcrtc
->crtc
, stat
);
529 /* Mask out those interrupts we haven't enabled */
530 v
= stat
& dcrtc
->irq_ena
;
532 if (v
& (VSYNC_IRQ
|GRA_FRAME_IRQ
|DUMB_FRAMEDONE
)) {
533 armada_drm_crtc_irq(dcrtc
, stat
);
539 static uint32_t armada_drm_crtc_calculate_csc(struct armada_crtc
*dcrtc
)
541 struct drm_display_mode
*adj
= &dcrtc
->crtc
.mode
;
544 if (dcrtc
->csc_yuv_mode
== CSC_YUV_CCIR709
)
545 val
|= CFG_CSC_YUV_CCIR709
;
546 if (dcrtc
->csc_rgb_mode
== CSC_RGB_STUDIO
)
547 val
|= CFG_CSC_RGB_STUDIO
;
550 * In auto mode, set the colorimetry, based upon the HDMI spec.
551 * 1280x720p, 1920x1080p and 1920x1080i use ITU709, others use
552 * ITU601. It may be more appropriate to set this depending on
553 * the source - but what if the graphic frame is YUV and the
554 * video frame is RGB?
556 if ((adj
->hdisplay
== 1280 && adj
->vdisplay
== 720 &&
557 !(adj
->flags
& DRM_MODE_FLAG_INTERLACE
)) ||
558 (adj
->hdisplay
== 1920 && adj
->vdisplay
== 1080)) {
559 if (dcrtc
->csc_yuv_mode
== CSC_AUTO
)
560 val
|= CFG_CSC_YUV_CCIR709
;
564 * We assume we're connected to a TV-like device, so the YUV->RGB
565 * conversion should produce a limited range. We should set this
566 * depending on the connectors attached to this CRTC, and what
567 * kind of device they report being connected.
569 if (dcrtc
->csc_rgb_mode
== CSC_AUTO
)
570 val
|= CFG_CSC_RGB_STUDIO
;
575 static void armada_drm_gra_plane_regs(struct armada_regs
*regs
,
576 struct drm_framebuffer
*fb
, struct armada_plane_state
*state
,
577 int x
, int y
, bool interlaced
)
582 i
= armada_drm_crtc_calc_fb(fb
, x
, y
, regs
, interlaced
);
583 armada_reg_queue_set(regs
, i
, state
->dst_yx
, LCD_SPU_GRA_OVSA_HPXL_VLN
);
584 armada_reg_queue_set(regs
, i
, state
->src_hw
, LCD_SPU_GRA_HPXL_VLN
);
585 armada_reg_queue_set(regs
, i
, state
->dst_hw
, LCD_SPU_GZM_HPXL_VLN
);
587 ctrl0
= state
->ctrl0
;
589 ctrl0
|= CFG_GRA_FTOGGLE
;
591 armada_reg_queue_mod(regs
, i
, ctrl0
, CFG_GRAFORMAT
|
592 CFG_GRA_MOD(CFG_SWAPRB
| CFG_SWAPUV
|
593 CFG_SWAPYU
| CFG_YUV2RGB
) |
594 CFG_PALETTE_ENA
| CFG_GRA_FTOGGLE
|
595 CFG_GRA_HSMOOTH
| CFG_GRA_ENA
,
597 armada_reg_queue_end(regs
, i
);
600 static void armada_drm_primary_set(struct drm_crtc
*crtc
,
601 struct drm_plane
*plane
, int x
, int y
)
603 struct armada_plane_state
*state
= &drm_to_armada_plane(plane
)->state
;
604 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
605 struct armada_regs regs
[8];
606 bool interlaced
= dcrtc
->interlaced
;
608 armada_drm_gra_plane_regs(regs
, plane
->fb
, state
, x
, y
, interlaced
);
609 armada_drm_crtc_update_regs(dcrtc
, regs
);
612 /* The mode_config.mutex will be held for this call */
613 static int armada_drm_crtc_mode_set(struct drm_crtc
*crtc
,
614 struct drm_display_mode
*mode
, struct drm_display_mode
*adj
,
615 int x
, int y
, struct drm_framebuffer
*old_fb
)
617 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
618 struct armada_regs regs
[17];
619 uint32_t lm
, rm
, tm
, bm
, val
, sclk
;
624 drm_framebuffer_get(crtc
->primary
->fb
);
626 interlaced
= !!(adj
->flags
& DRM_MODE_FLAG_INTERLACE
);
629 val
|= CFG_GRA_FMT(drm_fb_to_armada_fb(dcrtc
->crtc
.primary
->fb
)->fmt
);
630 val
|= CFG_GRA_MOD(drm_fb_to_armada_fb(dcrtc
->crtc
.primary
->fb
)->mod
);
632 if (drm_fb_to_armada_fb(dcrtc
->crtc
.primary
->fb
)->fmt
> CFG_420
)
633 val
|= CFG_PALETTE_ENA
;
635 drm_to_armada_plane(crtc
->primary
)->state
.ctrl0
= val
;
636 drm_to_armada_plane(crtc
->primary
)->state
.src_hw
=
637 drm_to_armada_plane(crtc
->primary
)->state
.dst_hw
=
638 adj
->crtc_vdisplay
<< 16 | adj
->crtc_hdisplay
;
639 drm_to_armada_plane(crtc
->primary
)->state
.dst_yx
= 0;
642 rm
= adj
->crtc_hsync_start
- adj
->crtc_hdisplay
;
643 lm
= adj
->crtc_htotal
- adj
->crtc_hsync_end
;
644 bm
= adj
->crtc_vsync_start
- adj
->crtc_vdisplay
;
645 tm
= adj
->crtc_vtotal
- adj
->crtc_vsync_end
;
647 DRM_DEBUG_DRIVER("H: %d %d %d %d lm %d rm %d\n",
649 adj
->crtc_hsync_start
,
651 adj
->crtc_htotal
, lm
, rm
);
652 DRM_DEBUG_DRIVER("V: %d %d %d %d tm %d bm %d\n",
654 adj
->crtc_vsync_start
,
656 adj
->crtc_vtotal
, tm
, bm
);
658 /* Wait for pending flips to complete */
659 armada_drm_plane_work_wait(drm_to_armada_plane(dcrtc
->crtc
.primary
),
660 MAX_SCHEDULE_TIMEOUT
);
662 drm_crtc_vblank_off(crtc
);
664 val
= dcrtc
->dumb_ctrl
& ~CFG_DUMB_ENA
;
665 if (val
!= dcrtc
->dumb_ctrl
) {
666 dcrtc
->dumb_ctrl
= val
;
667 writel_relaxed(val
, dcrtc
->base
+ LCD_SPU_DUMB_CTRL
);
671 * If we are blanked, we would have disabled the clock. Re-enable
672 * it so that compute_clock() does the right thing.
674 if (!IS_ERR(dcrtc
->clk
) && dpms_blanked(dcrtc
->dpms
))
675 WARN_ON(clk_prepare_enable(dcrtc
->clk
));
677 /* Now compute the divider for real */
678 dcrtc
->variant
->compute_clock(dcrtc
, adj
, &sclk
);
680 armada_reg_queue_set(regs
, i
, sclk
, LCD_CFG_SCLK_DIV
);
682 if (interlaced
^ dcrtc
->interlaced
) {
683 if (adj
->flags
& DRM_MODE_FLAG_INTERLACE
)
684 drm_crtc_vblank_get(&dcrtc
->crtc
);
686 drm_crtc_vblank_put(&dcrtc
->crtc
);
687 dcrtc
->interlaced
= interlaced
;
690 spin_lock_irqsave(&dcrtc
->irq_lock
, flags
);
692 /* Ensure graphic fifo is enabled */
693 armada_reg_queue_mod(regs
, i
, 0, CFG_PDWN64x66
, LCD_SPU_SRAM_PARA1
);
695 /* Even interlaced/progressive frame */
696 dcrtc
->v
[1].spu_v_h_total
= adj
->crtc_vtotal
<< 16 |
698 dcrtc
->v
[1].spu_v_porch
= tm
<< 16 | bm
;
699 val
= adj
->crtc_hsync_start
;
700 dcrtc
->v
[1].spu_adv_reg
= val
<< 20 | val
| ADV_VSYNCOFFEN
|
701 dcrtc
->variant
->spu_adv_reg
;
704 /* Odd interlaced frame */
705 dcrtc
->v
[0].spu_v_h_total
= dcrtc
->v
[1].spu_v_h_total
+
707 dcrtc
->v
[0].spu_v_porch
= dcrtc
->v
[1].spu_v_porch
+ 1;
708 val
= adj
->crtc_hsync_start
- adj
->crtc_htotal
/ 2;
709 dcrtc
->v
[0].spu_adv_reg
= val
<< 20 | val
| ADV_VSYNCOFFEN
|
710 dcrtc
->variant
->spu_adv_reg
;
712 dcrtc
->v
[0] = dcrtc
->v
[1];
715 val
= adj
->crtc_vdisplay
<< 16 | adj
->crtc_hdisplay
;
717 armada_reg_queue_set(regs
, i
, val
, LCD_SPU_V_H_ACTIVE
);
718 armada_reg_queue_set(regs
, i
, (lm
<< 16) | rm
, LCD_SPU_H_PORCH
);
719 armada_reg_queue_set(regs
, i
, dcrtc
->v
[0].spu_v_porch
, LCD_SPU_V_PORCH
);
720 armada_reg_queue_set(regs
, i
, dcrtc
->v
[0].spu_v_h_total
,
723 if (dcrtc
->variant
->has_spu_adv_reg
) {
724 armada_reg_queue_mod(regs
, i
, dcrtc
->v
[0].spu_adv_reg
,
725 ADV_VSYNC_L_OFF
| ADV_VSYNC_H_OFF
|
726 ADV_VSYNCOFFEN
, LCD_SPU_ADV_REG
);
729 val
= adj
->flags
& DRM_MODE_FLAG_NVSYNC
? CFG_VSYNC_INV
: 0;
730 armada_reg_queue_mod(regs
, i
, val
, CFG_VSYNC_INV
, LCD_SPU_DMA_CTRL1
);
732 val
= dcrtc
->spu_iopad_ctrl
| armada_drm_crtc_calculate_csc(dcrtc
);
733 armada_reg_queue_set(regs
, i
, val
, LCD_SPU_IOPAD_CONTROL
);
734 armada_reg_queue_end(regs
, i
);
736 armada_drm_crtc_update_regs(dcrtc
, regs
);
738 armada_drm_primary_set(crtc
, crtc
->primary
, x
, y
);
739 spin_unlock_irqrestore(&dcrtc
->irq_lock
, flags
);
741 armada_drm_crtc_update(dcrtc
);
743 drm_crtc_vblank_on(crtc
);
744 armada_drm_crtc_finish_fb(dcrtc
, old_fb
, dpms_blanked(dcrtc
->dpms
));
749 /* The mode_config.mutex will be held for this call */
750 static int armada_drm_crtc_mode_set_base(struct drm_crtc
*crtc
, int x
, int y
,
751 struct drm_framebuffer
*old_fb
)
753 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
754 struct armada_regs regs
[4];
757 i
= armada_drm_crtc_calc_fb(crtc
->primary
->fb
, crtc
->x
, crtc
->y
, regs
,
759 armada_reg_queue_end(regs
, i
);
761 /* Wait for pending flips to complete */
762 armada_drm_plane_work_wait(drm_to_armada_plane(dcrtc
->crtc
.primary
),
763 MAX_SCHEDULE_TIMEOUT
);
765 /* Take a reference to the new fb as we're using it */
766 drm_framebuffer_get(crtc
->primary
->fb
);
768 /* Update the base in the CRTC */
769 armada_drm_crtc_update_regs(dcrtc
, regs
);
771 /* Drop our previously held reference */
772 armada_drm_crtc_finish_fb(dcrtc
, old_fb
, dpms_blanked(dcrtc
->dpms
));
777 /* The mode_config.mutex will be held for this call */
778 static void armada_drm_crtc_disable(struct drm_crtc
*crtc
)
780 armada_drm_crtc_dpms(crtc
, DRM_MODE_DPMS_OFF
);
782 /* Disable our primary plane when we disable the CRTC. */
783 crtc
->primary
->funcs
->disable_plane(crtc
->primary
, NULL
);
786 static const struct drm_crtc_helper_funcs armada_crtc_helper_funcs
= {
787 .dpms
= armada_drm_crtc_dpms
,
788 .prepare
= armada_drm_crtc_prepare
,
789 .commit
= armada_drm_crtc_commit
,
790 .mode_fixup
= armada_drm_crtc_mode_fixup
,
791 .mode_set
= armada_drm_crtc_mode_set
,
792 .mode_set_base
= armada_drm_crtc_mode_set_base
,
793 .disable
= armada_drm_crtc_disable
,
796 static void armada_load_cursor_argb(void __iomem
*base
, uint32_t *pix
,
797 unsigned stride
, unsigned width
, unsigned height
)
802 addr
= SRAM_HWC32_RAM1
;
803 for (y
= 0; y
< height
; y
++) {
804 uint32_t *p
= &pix
[y
* stride
];
807 for (x
= 0; x
< width
; x
++, p
++) {
810 val
= (val
& 0xff00ff00) |
811 (val
& 0x000000ff) << 16 |
812 (val
& 0x00ff0000) >> 16;
815 base
+ LCD_SPU_SRAM_WRDAT
);
816 writel_relaxed(addr
| SRAM_WRITE
,
817 base
+ LCD_SPU_SRAM_CTRL
);
818 readl_relaxed(base
+ LCD_SPU_HWC_OVSA_HPXL_VLN
);
820 if ((addr
& 0x00ff) == 0)
822 if ((addr
& 0x30ff) == 0)
823 addr
= SRAM_HWC32_RAM2
;
828 static void armada_drm_crtc_cursor_tran(void __iomem
*base
)
832 for (addr
= 0; addr
< 256; addr
++) {
833 /* write the default value */
834 writel_relaxed(0x55555555, base
+ LCD_SPU_SRAM_WRDAT
);
835 writel_relaxed(addr
| SRAM_WRITE
| SRAM_HWC32_TRAN
,
836 base
+ LCD_SPU_SRAM_CTRL
);
840 static int armada_drm_crtc_cursor_update(struct armada_crtc
*dcrtc
, bool reload
)
842 uint32_t xoff
, xscr
, w
= dcrtc
->cursor_w
, s
;
843 uint32_t yoff
, yscr
, h
= dcrtc
->cursor_h
;
847 * Calculate the visible width and height of the cursor,
848 * screen position, and the position in the cursor bitmap.
850 if (dcrtc
->cursor_x
< 0) {
851 xoff
= -dcrtc
->cursor_x
;
854 } else if (dcrtc
->cursor_x
+ w
> dcrtc
->crtc
.mode
.hdisplay
) {
856 xscr
= dcrtc
->cursor_x
;
857 w
= max_t(int, dcrtc
->crtc
.mode
.hdisplay
- dcrtc
->cursor_x
, 0);
860 xscr
= dcrtc
->cursor_x
;
863 if (dcrtc
->cursor_y
< 0) {
864 yoff
= -dcrtc
->cursor_y
;
867 } else if (dcrtc
->cursor_y
+ h
> dcrtc
->crtc
.mode
.vdisplay
) {
869 yscr
= dcrtc
->cursor_y
;
870 h
= max_t(int, dcrtc
->crtc
.mode
.vdisplay
- dcrtc
->cursor_y
, 0);
873 yscr
= dcrtc
->cursor_y
;
876 /* On interlaced modes, the vertical cursor size must be halved */
878 if (dcrtc
->interlaced
) {
884 if (!dcrtc
->cursor_obj
|| !h
|| !w
) {
885 spin_lock_irq(&dcrtc
->irq_lock
);
886 armada_drm_crtc_disable_irq(dcrtc
, DUMB_FRAMEDONE_ENA
);
887 dcrtc
->cursor_update
= false;
888 armada_updatel(0, CFG_HWC_ENA
, dcrtc
->base
+ LCD_SPU_DMA_CTRL0
);
889 spin_unlock_irq(&dcrtc
->irq_lock
);
893 spin_lock_irq(&dcrtc
->irq_lock
);
894 para1
= readl_relaxed(dcrtc
->base
+ LCD_SPU_SRAM_PARA1
);
895 armada_updatel(CFG_CSB_256x32
, CFG_CSB_256x32
| CFG_PDWN256x32
,
896 dcrtc
->base
+ LCD_SPU_SRAM_PARA1
);
897 spin_unlock_irq(&dcrtc
->irq_lock
);
900 * Initialize the transparency if the SRAM was powered down.
901 * We must also reload the cursor data as well.
903 if (!(para1
& CFG_CSB_256x32
)) {
904 armada_drm_crtc_cursor_tran(dcrtc
->base
);
908 if (dcrtc
->cursor_hw_sz
!= (h
<< 16 | w
)) {
909 spin_lock_irq(&dcrtc
->irq_lock
);
910 armada_drm_crtc_disable_irq(dcrtc
, DUMB_FRAMEDONE_ENA
);
911 dcrtc
->cursor_update
= false;
912 armada_updatel(0, CFG_HWC_ENA
, dcrtc
->base
+ LCD_SPU_DMA_CTRL0
);
913 spin_unlock_irq(&dcrtc
->irq_lock
);
917 struct armada_gem_object
*obj
= dcrtc
->cursor_obj
;
919 /* Set the top-left corner of the cursor image */
921 pix
+= yoff
* s
+ xoff
;
922 armada_load_cursor_argb(dcrtc
->base
, pix
, s
, w
, h
);
925 /* Reload the cursor position, size and enable in the IRQ handler */
926 spin_lock_irq(&dcrtc
->irq_lock
);
927 dcrtc
->cursor_hw_pos
= yscr
<< 16 | xscr
;
928 dcrtc
->cursor_hw_sz
= h
<< 16 | w
;
929 dcrtc
->cursor_update
= true;
930 armada_drm_crtc_enable_irq(dcrtc
, DUMB_FRAMEDONE_ENA
);
931 spin_unlock_irq(&dcrtc
->irq_lock
);
936 static void cursor_update(void *data
)
938 armada_drm_crtc_cursor_update(data
, true);
941 static int armada_drm_crtc_cursor_set(struct drm_crtc
*crtc
,
942 struct drm_file
*file
, uint32_t handle
, uint32_t w
, uint32_t h
)
944 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
945 struct armada_gem_object
*obj
= NULL
;
948 /* If no cursor support, replicate drm's return value */
949 if (!dcrtc
->variant
->has_spu_adv_reg
)
952 if (handle
&& w
> 0 && h
> 0) {
953 /* maximum size is 64x32 or 32x64 */
954 if (w
> 64 || h
> 64 || (w
> 32 && h
> 32))
957 obj
= armada_gem_object_lookup(file
, handle
);
961 /* Must be a kernel-mapped object */
963 drm_gem_object_put_unlocked(&obj
->obj
);
967 if (obj
->obj
.size
< w
* h
* 4) {
968 DRM_ERROR("buffer is too small\n");
969 drm_gem_object_put_unlocked(&obj
->obj
);
974 if (dcrtc
->cursor_obj
) {
975 dcrtc
->cursor_obj
->update
= NULL
;
976 dcrtc
->cursor_obj
->update_data
= NULL
;
977 drm_gem_object_put_unlocked(&dcrtc
->cursor_obj
->obj
);
979 dcrtc
->cursor_obj
= obj
;
982 ret
= armada_drm_crtc_cursor_update(dcrtc
, true);
984 obj
->update_data
= dcrtc
;
985 obj
->update
= cursor_update
;
991 static int armada_drm_crtc_cursor_move(struct drm_crtc
*crtc
, int x
, int y
)
993 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
996 /* If no cursor support, replicate drm's return value */
997 if (!dcrtc
->variant
->has_spu_adv_reg
)
1000 dcrtc
->cursor_x
= x
;
1001 dcrtc
->cursor_y
= y
;
1002 ret
= armada_drm_crtc_cursor_update(dcrtc
, false);
1007 static void armada_drm_crtc_destroy(struct drm_crtc
*crtc
)
1009 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
1010 struct armada_private
*priv
= crtc
->dev
->dev_private
;
1012 if (dcrtc
->cursor_obj
)
1013 drm_gem_object_put_unlocked(&dcrtc
->cursor_obj
->obj
);
1015 priv
->dcrtc
[dcrtc
->num
] = NULL
;
1016 drm_crtc_cleanup(&dcrtc
->crtc
);
1018 if (!IS_ERR(dcrtc
->clk
))
1019 clk_disable_unprepare(dcrtc
->clk
);
1021 writel_relaxed(0, dcrtc
->base
+ LCD_SPU_IRQ_ENA
);
1023 of_node_put(dcrtc
->crtc
.port
);
1029 * The mode_config lock is held here, to prevent races between this
1032 static int armada_drm_crtc_page_flip(struct drm_crtc
*crtc
,
1033 struct drm_framebuffer
*fb
, struct drm_pending_vblank_event
*event
, uint32_t page_flip_flags
,
1034 struct drm_modeset_acquire_ctx
*ctx
)
1036 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
1037 struct armada_plane_work
*work
;
1041 /* We don't support changing the pixel format */
1042 if (fb
->format
!= crtc
->primary
->fb
->format
)
1045 work
= armada_drm_crtc_alloc_plane_work(dcrtc
->crtc
.primary
);
1049 work
->event
= event
;
1050 work
->old_fb
= dcrtc
->crtc
.primary
->fb
;
1052 i
= armada_drm_crtc_calc_fb(fb
, crtc
->x
, crtc
->y
, work
->regs
,
1054 armada_reg_queue_end(work
->regs
, i
);
1057 * Ensure that we hold a reference on the new framebuffer.
1058 * This has to match the behaviour in mode_set.
1060 drm_framebuffer_get(fb
);
1062 ret
= armada_drm_plane_work_queue(dcrtc
, work
);
1064 /* Undo our reference above */
1065 drm_framebuffer_put(fb
);
1071 * Don't take a reference on the new framebuffer;
1072 * drm_mode_page_flip_ioctl() has already grabbed a reference and
1073 * will _not_ drop that reference on successful return from this
1074 * function. Simply mark this new framebuffer as the current one.
1076 dcrtc
->crtc
.primary
->fb
= fb
;
1079 * Finally, if the display is blanked, we won't receive an
1080 * interrupt, so complete it now.
1082 if (dpms_blanked(dcrtc
->dpms
))
1083 armada_drm_plane_work_run(dcrtc
, dcrtc
->crtc
.primary
);
1089 armada_drm_crtc_set_property(struct drm_crtc
*crtc
,
1090 struct drm_property
*property
, uint64_t val
)
1092 struct armada_private
*priv
= crtc
->dev
->dev_private
;
1093 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
1094 bool update_csc
= false;
1096 if (property
== priv
->csc_yuv_prop
) {
1097 dcrtc
->csc_yuv_mode
= val
;
1099 } else if (property
== priv
->csc_rgb_prop
) {
1100 dcrtc
->csc_rgb_mode
= val
;
1107 val
= dcrtc
->spu_iopad_ctrl
|
1108 armada_drm_crtc_calculate_csc(dcrtc
);
1109 writel_relaxed(val
, dcrtc
->base
+ LCD_SPU_IOPAD_CONTROL
);
1115 /* These are called under the vbl_lock. */
1116 static int armada_drm_crtc_enable_vblank(struct drm_crtc
*crtc
)
1118 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
1120 armada_drm_crtc_enable_irq(dcrtc
, VSYNC_IRQ_ENA
);
1124 static void armada_drm_crtc_disable_vblank(struct drm_crtc
*crtc
)
1126 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
1128 armada_drm_crtc_disable_irq(dcrtc
, VSYNC_IRQ_ENA
);
1131 static const struct drm_crtc_funcs armada_crtc_funcs
= {
1132 .cursor_set
= armada_drm_crtc_cursor_set
,
1133 .cursor_move
= armada_drm_crtc_cursor_move
,
1134 .destroy
= armada_drm_crtc_destroy
,
1135 .set_config
= drm_crtc_helper_set_config
,
1136 .page_flip
= armada_drm_crtc_page_flip
,
1137 .set_property
= armada_drm_crtc_set_property
,
1138 .enable_vblank
= armada_drm_crtc_enable_vblank
,
1139 .disable_vblank
= armada_drm_crtc_disable_vblank
,
1142 static void armada_drm_primary_update_state(struct drm_plane_state
*state
,
1143 struct armada_regs
*regs
)
1145 struct armada_plane
*dplane
= drm_to_armada_plane(state
->plane
);
1146 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(state
->crtc
);
1147 struct armada_framebuffer
*dfb
= drm_fb_to_armada_fb(state
->fb
);
1149 unsigned int idx
= 0;
1152 val
= CFG_GRA_FMT(dfb
->fmt
) | CFG_GRA_MOD(dfb
->mod
);
1153 if (dfb
->fmt
> CFG_420
)
1154 val
|= CFG_PALETTE_ENA
;
1157 if (drm_rect_width(&state
->src
) >> 16 != drm_rect_width(&state
->dst
))
1158 val
|= CFG_GRA_HSMOOTH
;
1160 was_disabled
= !(dplane
->state
.ctrl0
& CFG_GRA_ENA
);
1162 armada_reg_queue_mod(regs
, idx
,
1163 0, CFG_PDWN64x66
, LCD_SPU_SRAM_PARA1
);
1165 dplane
->state
.ctrl0
= val
;
1166 dplane
->state
.src_hw
= (drm_rect_height(&state
->src
) & 0xffff0000) |
1167 drm_rect_width(&state
->src
) >> 16;
1168 dplane
->state
.dst_hw
= drm_rect_height(&state
->dst
) << 16 |
1169 drm_rect_width(&state
->dst
);
1170 dplane
->state
.dst_yx
= state
->dst
.y1
<< 16 | state
->dst
.x1
;
1172 armada_drm_gra_plane_regs(regs
+ idx
, &dfb
->fb
, &dplane
->state
,
1173 state
->src
.x1
>> 16, state
->src
.y1
>> 16,
1176 dplane
->state
.vsync_update
= !was_disabled
;
1177 dplane
->state
.changed
= true;
1180 static int armada_drm_primary_update(struct drm_plane
*plane
,
1181 struct drm_crtc
*crtc
, struct drm_framebuffer
*fb
,
1182 int crtc_x
, int crtc_y
, unsigned int crtc_w
, unsigned int crtc_h
,
1183 uint32_t src_x
, uint32_t src_y
, uint32_t src_w
, uint32_t src_h
,
1184 struct drm_modeset_acquire_ctx
*ctx
)
1186 struct armada_plane
*dplane
= drm_to_armada_plane(plane
);
1187 struct armada_crtc
*dcrtc
= drm_to_armada_crtc(crtc
);
1188 struct armada_plane_work
*work
;
1189 struct drm_plane_state state
= {
1201 .rotation
= DRM_MODE_ROTATE_0
,
1203 const struct drm_rect clip
= {
1204 .x2
= crtc
->mode
.hdisplay
,
1205 .y2
= crtc
->mode
.vdisplay
,
1209 ret
= drm_atomic_helper_check_plane_state(&state
, crtc
->state
, &clip
, 0,
1210 INT_MAX
, true, false);
1214 work
= &dplane
->works
[dplane
->next_work
];
1215 work
->fn
= armada_drm_crtc_complete_frame_work
;
1217 if (plane
->fb
!= fb
) {
1219 * Take a reference on the new framebuffer - we want to
1220 * hold on to it while the hardware is displaying it.
1222 drm_framebuffer_reference(fb
);
1224 work
->old_fb
= plane
->fb
;
1226 work
->old_fb
= NULL
;
1229 armada_drm_primary_update_state(&state
, work
->regs
);
1231 if (!dplane
->state
.changed
)
1234 /* Wait for pending work to complete */
1235 if (armada_drm_plane_work_wait(dplane
, HZ
/ 10) == 0)
1236 armada_drm_plane_work_cancel(dcrtc
, dplane
);
1238 if (!dplane
->state
.vsync_update
) {
1239 work
->fn(dcrtc
, work
);
1241 drm_framebuffer_unreference(work
->old_fb
);
1245 /* Queue it for update on the next interrupt if we are enabled */
1246 ret
= armada_drm_plane_work_queue(dcrtc
, work
);
1248 work
->fn(dcrtc
, work
);
1250 drm_framebuffer_unreference(work
->old_fb
);
1253 dplane
->next_work
= !dplane
->next_work
;
1258 int armada_drm_plane_disable(struct drm_plane
*plane
,
1259 struct drm_modeset_acquire_ctx
*ctx
)
1261 struct armada_plane
*dplane
= drm_to_armada_plane(plane
);
1262 struct armada_crtc
*dcrtc
;
1263 struct armada_plane_work
*work
;
1264 unsigned int idx
= 0;
1265 u32 sram_para1
, enable_mask
;
1271 * Arrange to power down most RAMs and FIFOs if this is the primary
1272 * plane, otherwise just the YUV FIFOs for the overlay plane.
1274 if (plane
->type
== DRM_PLANE_TYPE_PRIMARY
) {
1275 sram_para1
= CFG_PDWN256x32
| CFG_PDWN256x24
| CFG_PDWN256x8
|
1276 CFG_PDWN32x32
| CFG_PDWN64x66
;
1277 enable_mask
= CFG_GRA_ENA
;
1279 sram_para1
= CFG_PDWN16x66
| CFG_PDWN32x66
;
1280 enable_mask
= CFG_DMA_ENA
;
1283 dplane
->state
.ctrl0
&= ~enable_mask
;
1285 dcrtc
= drm_to_armada_crtc(plane
->crtc
);
1288 * Try to disable the plane and drop our ref on the framebuffer
1289 * at the next frame update. If we fail for any reason, disable
1290 * the plane immediately.
1292 work
= &dplane
->works
[dplane
->next_work
];
1293 work
->fn
= armada_drm_crtc_complete_disable_work
;
1294 work
->cancel
= armada_drm_crtc_complete_disable_work
;
1295 work
->old_fb
= plane
->fb
;
1297 armada_reg_queue_mod(work
->regs
, idx
,
1298 0, enable_mask
, LCD_SPU_DMA_CTRL0
);
1299 armada_reg_queue_mod(work
->regs
, idx
,
1300 sram_para1
, 0, LCD_SPU_SRAM_PARA1
);
1301 armada_reg_queue_end(work
->regs
, idx
);
1303 /* Wait for any preceding work to complete, but don't wedge */
1304 if (WARN_ON(!armada_drm_plane_work_wait(dplane
, HZ
)))
1305 armada_drm_plane_work_cancel(dcrtc
, dplane
);
1307 if (armada_drm_plane_work_queue(dcrtc
, work
)) {
1308 work
->fn(dcrtc
, work
);
1310 drm_framebuffer_unreference(work
->old_fb
);
1313 dplane
->next_work
= !dplane
->next_work
;
1318 static const struct drm_plane_funcs armada_primary_plane_funcs
= {
1319 .update_plane
= armada_drm_primary_update
,
1320 .disable_plane
= armada_drm_plane_disable
,
1321 .destroy
= drm_primary_helper_destroy
,
1324 int armada_drm_plane_init(struct armada_plane
*plane
)
1328 for (i
= 0; i
< ARRAY_SIZE(plane
->works
); i
++)
1329 plane
->works
[i
].plane
= &plane
->base
;
1331 init_waitqueue_head(&plane
->frame_wait
);
1336 static const struct drm_prop_enum_list armada_drm_csc_yuv_enum_list
[] = {
1337 { CSC_AUTO
, "Auto" },
1338 { CSC_YUV_CCIR601
, "CCIR601" },
1339 { CSC_YUV_CCIR709
, "CCIR709" },
1342 static const struct drm_prop_enum_list armada_drm_csc_rgb_enum_list
[] = {
1343 { CSC_AUTO
, "Auto" },
1344 { CSC_RGB_COMPUTER
, "Computer system" },
1345 { CSC_RGB_STUDIO
, "Studio" },
1348 static int armada_drm_crtc_create_properties(struct drm_device
*dev
)
1350 struct armada_private
*priv
= dev
->dev_private
;
1352 if (priv
->csc_yuv_prop
)
1355 priv
->csc_yuv_prop
= drm_property_create_enum(dev
, 0,
1356 "CSC_YUV", armada_drm_csc_yuv_enum_list
,
1357 ARRAY_SIZE(armada_drm_csc_yuv_enum_list
));
1358 priv
->csc_rgb_prop
= drm_property_create_enum(dev
, 0,
1359 "CSC_RGB", armada_drm_csc_rgb_enum_list
,
1360 ARRAY_SIZE(armada_drm_csc_rgb_enum_list
));
1362 if (!priv
->csc_yuv_prop
|| !priv
->csc_rgb_prop
)
1368 static int armada_drm_crtc_create(struct drm_device
*drm
, struct device
*dev
,
1369 struct resource
*res
, int irq
, const struct armada_variant
*variant
,
1370 struct device_node
*port
)
1372 struct armada_private
*priv
= drm
->dev_private
;
1373 struct armada_crtc
*dcrtc
;
1374 struct armada_plane
*primary
;
1378 ret
= armada_drm_crtc_create_properties(drm
);
1382 base
= devm_ioremap_resource(dev
, res
);
1384 return PTR_ERR(base
);
1386 dcrtc
= kzalloc(sizeof(*dcrtc
), GFP_KERNEL
);
1388 DRM_ERROR("failed to allocate Armada crtc\n");
1392 if (dev
!= drm
->dev
)
1393 dev_set_drvdata(dev
, dcrtc
);
1395 dcrtc
->variant
= variant
;
1397 dcrtc
->num
= drm
->mode_config
.num_crtc
;
1398 dcrtc
->clk
= ERR_PTR(-EINVAL
);
1399 dcrtc
->csc_yuv_mode
= CSC_AUTO
;
1400 dcrtc
->csc_rgb_mode
= CSC_AUTO
;
1401 dcrtc
->cfg_dumb_ctrl
= DUMB24_RGB888_0
;
1402 dcrtc
->spu_iopad_ctrl
= CFG_VSCALE_LN_EN
| CFG_IOPAD_DUMB24
;
1403 spin_lock_init(&dcrtc
->irq_lock
);
1404 dcrtc
->irq_ena
= CLEAN_SPU_IRQ_ISR
;
1406 /* Initialize some registers which we don't otherwise set */
1407 writel_relaxed(0x00000001, dcrtc
->base
+ LCD_CFG_SCLK_DIV
);
1408 writel_relaxed(0x00000000, dcrtc
->base
+ LCD_SPU_BLANKCOLOR
);
1409 writel_relaxed(dcrtc
->spu_iopad_ctrl
,
1410 dcrtc
->base
+ LCD_SPU_IOPAD_CONTROL
);
1411 writel_relaxed(0x00000000, dcrtc
->base
+ LCD_SPU_SRAM_PARA0
);
1412 writel_relaxed(CFG_PDWN256x32
| CFG_PDWN256x24
| CFG_PDWN256x8
|
1413 CFG_PDWN32x32
| CFG_PDWN16x66
| CFG_PDWN32x66
|
1414 CFG_PDWN64x66
, dcrtc
->base
+ LCD_SPU_SRAM_PARA1
);
1415 writel_relaxed(0x2032ff81, dcrtc
->base
+ LCD_SPU_DMA_CTRL1
);
1416 writel_relaxed(dcrtc
->irq_ena
, dcrtc
->base
+ LCD_SPU_IRQ_ENA
);
1417 writel_relaxed(0, dcrtc
->base
+ LCD_SPU_IRQ_ISR
);
1419 ret
= devm_request_irq(dev
, irq
, armada_drm_irq
, 0, "armada_drm_crtc",
1424 if (dcrtc
->variant
->init
) {
1425 ret
= dcrtc
->variant
->init(dcrtc
, dev
);
1430 /* Ensure AXI pipeline is enabled */
1431 armada_updatel(CFG_ARBFAST_ENA
, 0, dcrtc
->base
+ LCD_SPU_DMA_CTRL0
);
1433 priv
->dcrtc
[dcrtc
->num
] = dcrtc
;
1435 dcrtc
->crtc
.port
= port
;
1437 primary
= kzalloc(sizeof(*primary
), GFP_KERNEL
);
1443 ret
= armada_drm_plane_init(primary
);
1449 ret
= drm_universal_plane_init(drm
, &primary
->base
, 0,
1450 &armada_primary_plane_funcs
,
1451 armada_primary_formats
,
1452 ARRAY_SIZE(armada_primary_formats
),
1454 DRM_PLANE_TYPE_PRIMARY
, NULL
);
1460 ret
= drm_crtc_init_with_planes(drm
, &dcrtc
->crtc
, &primary
->base
, NULL
,
1461 &armada_crtc_funcs
, NULL
);
1465 drm_crtc_helper_add(&dcrtc
->crtc
, &armada_crtc_helper_funcs
);
1467 drm_object_attach_property(&dcrtc
->crtc
.base
, priv
->csc_yuv_prop
,
1468 dcrtc
->csc_yuv_mode
);
1469 drm_object_attach_property(&dcrtc
->crtc
.base
, priv
->csc_rgb_prop
,
1470 dcrtc
->csc_rgb_mode
);
1472 return armada_overlay_plane_create(drm
, 1 << dcrtc
->num
);
1475 primary
->base
.funcs
->destroy(&primary
->base
);
1483 armada_lcd_bind(struct device
*dev
, struct device
*master
, void *data
)
1485 struct platform_device
*pdev
= to_platform_device(dev
);
1486 struct drm_device
*drm
= data
;
1487 struct resource
*res
= platform_get_resource(pdev
, IORESOURCE_MEM
, 0);
1488 int irq
= platform_get_irq(pdev
, 0);
1489 const struct armada_variant
*variant
;
1490 struct device_node
*port
= NULL
;
1495 if (!dev
->of_node
) {
1496 const struct platform_device_id
*id
;
1498 id
= platform_get_device_id(pdev
);
1502 variant
= (const struct armada_variant
*)id
->driver_data
;
1504 const struct of_device_id
*match
;
1505 struct device_node
*np
, *parent
= dev
->of_node
;
1507 match
= of_match_device(dev
->driver
->of_match_table
, dev
);
1511 np
= of_get_child_by_name(parent
, "ports");
1514 port
= of_get_child_by_name(parent
, "port");
1517 dev_err(dev
, "no port node found in %pOF\n", parent
);
1521 variant
= match
->data
;
1524 return armada_drm_crtc_create(drm
, dev
, res
, irq
, variant
, port
);
1528 armada_lcd_unbind(struct device
*dev
, struct device
*master
, void *data
)
1530 struct armada_crtc
*dcrtc
= dev_get_drvdata(dev
);
1532 armada_drm_crtc_destroy(&dcrtc
->crtc
);
1535 static const struct component_ops armada_lcd_ops
= {
1536 .bind
= armada_lcd_bind
,
1537 .unbind
= armada_lcd_unbind
,
1540 static int armada_lcd_probe(struct platform_device
*pdev
)
1542 return component_add(&pdev
->dev
, &armada_lcd_ops
);
1545 static int armada_lcd_remove(struct platform_device
*pdev
)
1547 component_del(&pdev
->dev
, &armada_lcd_ops
);
1551 static const struct of_device_id armada_lcd_of_match
[] = {
1553 .compatible
= "marvell,dove-lcd",
1554 .data
= &armada510_ops
,
1558 MODULE_DEVICE_TABLE(of
, armada_lcd_of_match
);
1560 static const struct platform_device_id armada_lcd_platform_ids
[] = {
1562 .name
= "armada-lcd",
1563 .driver_data
= (unsigned long)&armada510_ops
,
1565 .name
= "armada-510-lcd",
1566 .driver_data
= (unsigned long)&armada510_ops
,
1570 MODULE_DEVICE_TABLE(platform
, armada_lcd_platform_ids
);
1572 struct platform_driver armada_lcd_platform_driver
= {
1573 .probe
= armada_lcd_probe
,
1574 .remove
= armada_lcd_remove
,
1576 .name
= "armada-lcd",
1577 .owner
= THIS_MODULE
,
1578 .of_match_table
= armada_lcd_of_match
,
1580 .id_table
= armada_lcd_platform_ids
,