armada_crtc.c 31 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (C) 2012 Russell King
  4. * Rewritten from the dovefb driver, and Armada510 manuals.
  5. */
  6. #include <linux/clk.h>
  7. #include <linux/component.h>
  8. #include <linux/module.h>
  9. #include <linux/of_device.h>
  10. #include <linux/platform_device.h>
  11. #include <drm/drm_atomic.h>
  12. #include <drm/drm_atomic_helper.h>
  13. #include <drm/drm_probe_helper.h>
  14. #include <drm/drm_vblank.h>
  15. #include "armada_crtc.h"
  16. #include "armada_drm.h"
  17. #include "armada_fb.h"
  18. #include "armada_gem.h"
  19. #include "armada_hw.h"
  20. #include "armada_plane.h"
  21. #include "armada_trace.h"
  22. /*
  23. * A note about interlacing. Let's consider HDMI 1920x1080i.
  24. * The timing parameters we have from X are:
  25. * Hact HsyA HsyI Htot Vact VsyA VsyI Vtot
  26. * 1920 2448 2492 2640 1080 1084 1094 1125
  27. * Which get translated to:
  28. * Hact HsyA HsyI Htot Vact VsyA VsyI Vtot
  29. * 1920 2448 2492 2640 540 542 547 562
  30. *
  31. * This is how it is defined by CEA-861-D - line and pixel numbers are
  32. * referenced to the rising edge of VSYNC and HSYNC. Total clocks per
  33. * line: 2640. The odd frame, the first active line is at line 21, and
  34. * the even frame, the first active line is 584.
  35. *
  36. * LN: 560 561 562 563 567 568 569
  37. * DE: ~~~|____________________________//__________________________
  38. * HSYNC: ____|~|_____|~|_____|~|_____|~|_//__|~|_____|~|_____|~|_____
  39. * VSYNC: _________________________|~~~~~~//~~~~~~~~~~~~~~~|__________
  40. * 22 blanking lines. VSYNC at 1320 (referenced to the HSYNC rising edge).
  41. *
  42. * LN: 1123 1124 1125 1 5 6 7
  43. * DE: ~~~|____________________________//__________________________
  44. * HSYNC: ____|~|_____|~|_____|~|_____|~|_//__|~|_____|~|_____|~|_____
  45. * VSYNC: ____________________|~~~~~~~~~~~//~~~~~~~~~~|_______________
  46. * 23 blanking lines
  47. *
  48. * The Armada LCD Controller line and pixel numbers are, like X timings,
  49. * referenced to the top left of the active frame.
  50. *
  51. * So, translating these to our LCD controller:
  52. * Odd frame, 563 total lines, VSYNC at line 543-548, pixel 1128.
  53. * Even frame, 562 total lines, VSYNC at line 542-547, pixel 2448.
  54. * Note: Vsync front porch remains constant!
  55. *
  56. * if (odd_frame) {
  57. * vtotal = mode->crtc_vtotal + 1;
  58. * vbackporch = mode->crtc_vsync_start - mode->crtc_vdisplay + 1;
  59. * vhorizpos = mode->crtc_hsync_start - mode->crtc_htotal / 2
  60. * } else {
  61. * vtotal = mode->crtc_vtotal;
  62. * vbackporch = mode->crtc_vsync_start - mode->crtc_vdisplay;
  63. * vhorizpos = mode->crtc_hsync_start;
  64. * }
  65. * vfrontporch = mode->crtc_vtotal - mode->crtc_vsync_end;
  66. *
  67. * So, we need to reprogram these registers on each vsync event:
  68. * LCD_SPU_V_PORCH, LCD_SPU_ADV_REG, LCD_SPUT_V_H_TOTAL
  69. *
  70. * Note: we do not use the frame done interrupts because these appear
  71. * to happen too early, and lead to jitter on the display (presumably
  72. * they occur at the end of the last active line, before the vsync back
  73. * porch, which we're reprogramming.)
  74. */
  75. void
  76. armada_drm_crtc_update_regs(struct armada_crtc *dcrtc, struct armada_regs *regs)
  77. {
  78. while (regs->offset != ~0) {
  79. void __iomem *reg = dcrtc->base + regs->offset;
  80. uint32_t val;
  81. val = regs->mask;
  82. if (val != 0)
  83. val &= readl_relaxed(reg);
  84. writel_relaxed(val | regs->val, reg);
  85. ++regs;
  86. }
  87. }
  88. static void armada_drm_crtc_update(struct armada_crtc *dcrtc, bool enable)
  89. {
  90. uint32_t dumb_ctrl;
  91. dumb_ctrl = dcrtc->cfg_dumb_ctrl;
  92. if (enable)
  93. dumb_ctrl |= CFG_DUMB_ENA;
  94. /*
  95. * When the dumb interface isn't in DUMB24_RGB888_0 mode, it might
  96. * be using SPI or GPIO. If we set this to DUMB_BLANK, we will
  97. * force LCD_D[23:0] to output blank color, overriding the GPIO or
  98. * SPI usage. So leave it as-is unless in DUMB24_RGB888_0 mode.
  99. */
  100. if (!enable && (dumb_ctrl & DUMB_MASK) == DUMB24_RGB888_0) {
  101. dumb_ctrl &= ~DUMB_MASK;
  102. dumb_ctrl |= DUMB_BLANK;
  103. }
  104. armada_updatel(dumb_ctrl,
  105. ~(CFG_INV_CSYNC | CFG_INV_HSYNC | CFG_INV_VSYNC),
  106. dcrtc->base + LCD_SPU_DUMB_CTRL);
  107. }
  108. static void armada_drm_crtc_queue_state_event(struct drm_crtc *crtc)
  109. {
  110. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  111. struct drm_pending_vblank_event *event;
  112. /* If we have an event, we need vblank events enabled */
  113. event = xchg(&crtc->state->event, NULL);
  114. if (event) {
  115. WARN_ON(drm_crtc_vblank_get(crtc) != 0);
  116. dcrtc->event = event;
  117. }
  118. }
  119. static void armada_drm_update_gamma(struct drm_crtc *crtc)
  120. {
  121. struct drm_property_blob *blob = crtc->state->gamma_lut;
  122. void __iomem *base = drm_to_armada_crtc(crtc)->base;
  123. int i;
  124. if (blob) {
  125. struct drm_color_lut *lut = blob->data;
  126. armada_updatel(CFG_CSB_256x8, CFG_CSB_256x8 | CFG_PDWN256x8,
  127. base + LCD_SPU_SRAM_PARA1);
  128. for (i = 0; i < 256; i++) {
  129. writel_relaxed(drm_color_lut_extract(lut[i].red, 8),
  130. base + LCD_SPU_SRAM_WRDAT);
  131. writel_relaxed(i | SRAM_WRITE | SRAM_GAMMA_YR,
  132. base + LCD_SPU_SRAM_CTRL);
  133. readl_relaxed(base + LCD_SPU_HWC_OVSA_HPXL_VLN);
  134. writel_relaxed(drm_color_lut_extract(lut[i].green, 8),
  135. base + LCD_SPU_SRAM_WRDAT);
  136. writel_relaxed(i | SRAM_WRITE | SRAM_GAMMA_UG,
  137. base + LCD_SPU_SRAM_CTRL);
  138. readl_relaxed(base + LCD_SPU_HWC_OVSA_HPXL_VLN);
  139. writel_relaxed(drm_color_lut_extract(lut[i].blue, 8),
  140. base + LCD_SPU_SRAM_WRDAT);
  141. writel_relaxed(i | SRAM_WRITE | SRAM_GAMMA_VB,
  142. base + LCD_SPU_SRAM_CTRL);
  143. readl_relaxed(base + LCD_SPU_HWC_OVSA_HPXL_VLN);
  144. }
  145. armada_updatel(CFG_GAMMA_ENA, CFG_GAMMA_ENA,
  146. base + LCD_SPU_DMA_CTRL0);
  147. } else {
  148. armada_updatel(0, CFG_GAMMA_ENA, base + LCD_SPU_DMA_CTRL0);
  149. armada_updatel(CFG_PDWN256x8, CFG_CSB_256x8 | CFG_PDWN256x8,
  150. base + LCD_SPU_SRAM_PARA1);
  151. }
  152. }
  153. static enum drm_mode_status armada_drm_crtc_mode_valid(struct drm_crtc *crtc,
  154. const struct drm_display_mode *mode)
  155. {
  156. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  157. if (mode->vscan > 1)
  158. return MODE_NO_VSCAN;
  159. if (mode->flags & DRM_MODE_FLAG_DBLSCAN)
  160. return MODE_NO_DBLESCAN;
  161. if (mode->flags & DRM_MODE_FLAG_HSKEW)
  162. return MODE_H_ILLEGAL;
  163. /* We can't do interlaced modes if we don't have the SPU_ADV_REG */
  164. if (!dcrtc->variant->has_spu_adv_reg &&
  165. mode->flags & DRM_MODE_FLAG_INTERLACE)
  166. return MODE_NO_INTERLACE;
  167. if (mode->flags & (DRM_MODE_FLAG_BCAST | DRM_MODE_FLAG_PIXMUX |
  168. DRM_MODE_FLAG_CLKDIV2))
  169. return MODE_BAD;
  170. return MODE_OK;
  171. }
  172. /* The mode_config.mutex will be held for this call */
  173. static bool armada_drm_crtc_mode_fixup(struct drm_crtc *crtc,
  174. const struct drm_display_mode *mode, struct drm_display_mode *adj)
  175. {
  176. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  177. int ret;
  178. /*
  179. * Set CRTC modesetting parameters for the adjusted mode. This is
  180. * applied after the connectors, bridges, and encoders have fixed up
  181. * this mode, as described above drm_atomic_helper_check_modeset().
  182. */
  183. drm_mode_set_crtcinfo(adj, CRTC_INTERLACE_HALVE_V);
  184. /*
  185. * Validate the adjusted mode in case an encoder/bridge has set
  186. * something we don't support.
  187. */
  188. if (armada_drm_crtc_mode_valid(crtc, adj) != MODE_OK)
  189. return false;
  190. /* Check whether the display mode is possible */
  191. ret = dcrtc->variant->compute_clock(dcrtc, adj, NULL);
  192. if (ret)
  193. return false;
  194. return true;
  195. }
  196. /* These are locked by dev->vbl_lock */
  197. static void armada_drm_crtc_disable_irq(struct armada_crtc *dcrtc, u32 mask)
  198. {
  199. if (dcrtc->irq_ena & mask) {
  200. dcrtc->irq_ena &= ~mask;
  201. writel(dcrtc->irq_ena, dcrtc->base + LCD_SPU_IRQ_ENA);
  202. }
  203. }
  204. static void armada_drm_crtc_enable_irq(struct armada_crtc *dcrtc, u32 mask)
  205. {
  206. if ((dcrtc->irq_ena & mask) != mask) {
  207. dcrtc->irq_ena |= mask;
  208. writel(dcrtc->irq_ena, dcrtc->base + LCD_SPU_IRQ_ENA);
  209. if (readl_relaxed(dcrtc->base + LCD_SPU_IRQ_ISR) & mask)
  210. writel(0, dcrtc->base + LCD_SPU_IRQ_ISR);
  211. }
  212. }
  213. static void armada_drm_crtc_irq(struct armada_crtc *dcrtc, u32 stat)
  214. {
  215. struct drm_pending_vblank_event *event;
  216. void __iomem *base = dcrtc->base;
  217. if (stat & DMA_FF_UNDERFLOW)
  218. DRM_ERROR("video underflow on crtc %u\n", dcrtc->num);
  219. if (stat & GRA_FF_UNDERFLOW)
  220. DRM_ERROR("graphics underflow on crtc %u\n", dcrtc->num);
  221. if (stat & VSYNC_IRQ)
  222. drm_crtc_handle_vblank(&dcrtc->crtc);
  223. spin_lock(&dcrtc->irq_lock);
  224. if (stat & GRA_FRAME_IRQ && dcrtc->interlaced) {
  225. int i = stat & GRA_FRAME_IRQ0 ? 0 : 1;
  226. uint32_t val;
  227. writel_relaxed(dcrtc->v[i].spu_v_porch, base + LCD_SPU_V_PORCH);
  228. writel_relaxed(dcrtc->v[i].spu_v_h_total,
  229. base + LCD_SPUT_V_H_TOTAL);
  230. val = readl_relaxed(base + LCD_SPU_ADV_REG);
  231. val &= ~(ADV_VSYNC_L_OFF | ADV_VSYNC_H_OFF | ADV_VSYNCOFFEN);
  232. val |= dcrtc->v[i].spu_adv_reg;
  233. writel_relaxed(val, base + LCD_SPU_ADV_REG);
  234. }
  235. if (stat & dcrtc->irq_ena & DUMB_FRAMEDONE) {
  236. if (dcrtc->update_pending) {
  237. armada_drm_crtc_update_regs(dcrtc, dcrtc->regs);
  238. dcrtc->update_pending = false;
  239. }
  240. if (dcrtc->cursor_update) {
  241. writel_relaxed(dcrtc->cursor_hw_pos,
  242. base + LCD_SPU_HWC_OVSA_HPXL_VLN);
  243. writel_relaxed(dcrtc->cursor_hw_sz,
  244. base + LCD_SPU_HWC_HPXL_VLN);
  245. armada_updatel(CFG_HWC_ENA,
  246. CFG_HWC_ENA | CFG_HWC_1BITMOD |
  247. CFG_HWC_1BITENA,
  248. base + LCD_SPU_DMA_CTRL0);
  249. dcrtc->cursor_update = false;
  250. }
  251. armada_drm_crtc_disable_irq(dcrtc, DUMB_FRAMEDONE_ENA);
  252. }
  253. spin_unlock(&dcrtc->irq_lock);
  254. if (stat & VSYNC_IRQ && !dcrtc->update_pending) {
  255. event = xchg(&dcrtc->event, NULL);
  256. if (event) {
  257. spin_lock(&dcrtc->crtc.dev->event_lock);
  258. drm_crtc_send_vblank_event(&dcrtc->crtc, event);
  259. spin_unlock(&dcrtc->crtc.dev->event_lock);
  260. drm_crtc_vblank_put(&dcrtc->crtc);
  261. }
  262. }
  263. }
  264. static irqreturn_t armada_drm_irq(int irq, void *arg)
  265. {
  266. struct armada_crtc *dcrtc = arg;
  267. u32 v, stat = readl_relaxed(dcrtc->base + LCD_SPU_IRQ_ISR);
  268. /*
  269. * Reading the ISR appears to clear bits provided CLEAN_SPU_IRQ_ISR
  270. * is set. Writing has some other effect to acknowledge the IRQ -
  271. * without this, we only get a single IRQ.
  272. */
  273. writel_relaxed(0, dcrtc->base + LCD_SPU_IRQ_ISR);
  274. trace_armada_drm_irq(&dcrtc->crtc, stat);
  275. /* Mask out those interrupts we haven't enabled */
  276. v = stat & dcrtc->irq_ena;
  277. if (v & (VSYNC_IRQ|GRA_FRAME_IRQ|DUMB_FRAMEDONE)) {
  278. armada_drm_crtc_irq(dcrtc, stat);
  279. return IRQ_HANDLED;
  280. }
  281. return IRQ_NONE;
  282. }
  283. /* The mode_config.mutex will be held for this call */
  284. static void armada_drm_crtc_mode_set_nofb(struct drm_crtc *crtc)
  285. {
  286. struct drm_display_mode *adj = &crtc->state->adjusted_mode;
  287. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  288. struct armada_regs regs[17];
  289. uint32_t lm, rm, tm, bm, val, sclk;
  290. unsigned long flags;
  291. unsigned i;
  292. bool interlaced = !!(adj->flags & DRM_MODE_FLAG_INTERLACE);
  293. i = 0;
  294. rm = adj->crtc_hsync_start - adj->crtc_hdisplay;
  295. lm = adj->crtc_htotal - adj->crtc_hsync_end;
  296. bm = adj->crtc_vsync_start - adj->crtc_vdisplay;
  297. tm = adj->crtc_vtotal - adj->crtc_vsync_end;
  298. DRM_DEBUG_KMS("[CRTC:%d:%s] mode " DRM_MODE_FMT "\n",
  299. crtc->base.id, crtc->name, DRM_MODE_ARG(adj));
  300. DRM_DEBUG_KMS("lm %d rm %d tm %d bm %d\n", lm, rm, tm, bm);
  301. /* Now compute the divider for real */
  302. dcrtc->variant->compute_clock(dcrtc, adj, &sclk);
  303. armada_reg_queue_set(regs, i, sclk, LCD_CFG_SCLK_DIV);
  304. spin_lock_irqsave(&dcrtc->irq_lock, flags);
  305. dcrtc->interlaced = interlaced;
  306. /* Even interlaced/progressive frame */
  307. dcrtc->v[1].spu_v_h_total = adj->crtc_vtotal << 16 |
  308. adj->crtc_htotal;
  309. dcrtc->v[1].spu_v_porch = tm << 16 | bm;
  310. val = adj->crtc_hsync_start;
  311. dcrtc->v[1].spu_adv_reg = val << 20 | val | ADV_VSYNCOFFEN;
  312. if (interlaced) {
  313. /* Odd interlaced frame */
  314. val -= adj->crtc_htotal / 2;
  315. dcrtc->v[0].spu_adv_reg = val << 20 | val | ADV_VSYNCOFFEN;
  316. dcrtc->v[0].spu_v_h_total = dcrtc->v[1].spu_v_h_total +
  317. (1 << 16);
  318. dcrtc->v[0].spu_v_porch = dcrtc->v[1].spu_v_porch + 1;
  319. } else {
  320. dcrtc->v[0] = dcrtc->v[1];
  321. }
  322. val = adj->crtc_vdisplay << 16 | adj->crtc_hdisplay;
  323. armada_reg_queue_set(regs, i, val, LCD_SPU_V_H_ACTIVE);
  324. armada_reg_queue_set(regs, i, (lm << 16) | rm, LCD_SPU_H_PORCH);
  325. armada_reg_queue_set(regs, i, dcrtc->v[0].spu_v_porch, LCD_SPU_V_PORCH);
  326. armada_reg_queue_set(regs, i, dcrtc->v[0].spu_v_h_total,
  327. LCD_SPUT_V_H_TOTAL);
  328. if (dcrtc->variant->has_spu_adv_reg)
  329. armada_reg_queue_mod(regs, i, dcrtc->v[0].spu_adv_reg,
  330. ADV_VSYNC_L_OFF | ADV_VSYNC_H_OFF |
  331. ADV_VSYNCOFFEN, LCD_SPU_ADV_REG);
  332. val = adj->flags & DRM_MODE_FLAG_NVSYNC ? CFG_VSYNC_INV : 0;
  333. armada_reg_queue_mod(regs, i, val, CFG_VSYNC_INV, LCD_SPU_DMA_CTRL1);
  334. /*
  335. * The documentation doesn't indicate what the normal state of
  336. * the sync signals are. Sebastian Hesselbart kindly probed
  337. * these signals on his board to determine their state.
  338. *
  339. * The non-inverted state of the sync signals is active high.
  340. * Setting these bits makes the appropriate signal active low.
  341. */
  342. val = 0;
  343. if (adj->flags & DRM_MODE_FLAG_NCSYNC)
  344. val |= CFG_INV_CSYNC;
  345. if (adj->flags & DRM_MODE_FLAG_NHSYNC)
  346. val |= CFG_INV_HSYNC;
  347. if (adj->flags & DRM_MODE_FLAG_NVSYNC)
  348. val |= CFG_INV_VSYNC;
  349. armada_reg_queue_mod(regs, i, val, CFG_INV_CSYNC | CFG_INV_HSYNC |
  350. CFG_INV_VSYNC, LCD_SPU_DUMB_CTRL);
  351. armada_reg_queue_end(regs, i);
  352. armada_drm_crtc_update_regs(dcrtc, regs);
  353. spin_unlock_irqrestore(&dcrtc->irq_lock, flags);
  354. }
  355. static int armada_drm_crtc_atomic_check(struct drm_crtc *crtc,
  356. struct drm_atomic_state *state)
  357. {
  358. struct drm_crtc_state *crtc_state = drm_atomic_get_new_crtc_state(state,
  359. crtc);
  360. DRM_DEBUG_KMS("[CRTC:%d:%s]\n", crtc->base.id, crtc->name);
  361. if (crtc_state->gamma_lut && drm_color_lut_size(crtc_state->gamma_lut) != 256)
  362. return -EINVAL;
  363. if (crtc_state->color_mgmt_changed)
  364. crtc_state->planes_changed = true;
  365. return 0;
  366. }
  367. static void armada_drm_crtc_atomic_begin(struct drm_crtc *crtc,
  368. struct drm_atomic_state *state)
  369. {
  370. struct drm_crtc_state *crtc_state = drm_atomic_get_new_crtc_state(state,
  371. crtc);
  372. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  373. DRM_DEBUG_KMS("[CRTC:%d:%s]\n", crtc->base.id, crtc->name);
  374. if (crtc_state->color_mgmt_changed)
  375. armada_drm_update_gamma(crtc);
  376. dcrtc->regs_idx = 0;
  377. dcrtc->regs = dcrtc->atomic_regs;
  378. }
  379. static void armada_drm_crtc_atomic_flush(struct drm_crtc *crtc,
  380. struct drm_atomic_state *state)
  381. {
  382. struct drm_crtc_state *crtc_state = drm_atomic_get_new_crtc_state(state,
  383. crtc);
  384. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  385. DRM_DEBUG_KMS("[CRTC:%d:%s]\n", crtc->base.id, crtc->name);
  386. armada_reg_queue_end(dcrtc->regs, dcrtc->regs_idx);
  387. /*
  388. * If we aren't doing a full modeset, then we need to queue
  389. * the event here.
  390. */
  391. if (!drm_atomic_crtc_needs_modeset(crtc_state)) {
  392. dcrtc->update_pending = true;
  393. armada_drm_crtc_queue_state_event(crtc);
  394. spin_lock_irq(&dcrtc->irq_lock);
  395. armada_drm_crtc_enable_irq(dcrtc, DUMB_FRAMEDONE_ENA);
  396. spin_unlock_irq(&dcrtc->irq_lock);
  397. } else {
  398. spin_lock_irq(&dcrtc->irq_lock);
  399. armada_drm_crtc_update_regs(dcrtc, dcrtc->regs);
  400. spin_unlock_irq(&dcrtc->irq_lock);
  401. }
  402. }
  403. static void armada_drm_crtc_atomic_disable(struct drm_crtc *crtc,
  404. struct drm_atomic_state *state)
  405. {
  406. struct drm_crtc_state *old_state = drm_atomic_get_old_crtc_state(state,
  407. crtc);
  408. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  409. struct drm_pending_vblank_event *event;
  410. DRM_DEBUG_KMS("[CRTC:%d:%s]\n", crtc->base.id, crtc->name);
  411. if (old_state->adjusted_mode.flags & DRM_MODE_FLAG_INTERLACE)
  412. drm_crtc_vblank_put(crtc);
  413. drm_crtc_vblank_off(crtc);
  414. armada_drm_crtc_update(dcrtc, false);
  415. if (!crtc->state->active) {
  416. /*
  417. * This modeset will be leaving the CRTC disabled, so
  418. * call the backend to disable upstream clocks etc.
  419. */
  420. if (dcrtc->variant->disable)
  421. dcrtc->variant->disable(dcrtc);
  422. /*
  423. * We will not receive any further vblank events.
  424. * Send the flip_done event manually.
  425. */
  426. event = crtc->state->event;
  427. crtc->state->event = NULL;
  428. if (event) {
  429. spin_lock_irq(&crtc->dev->event_lock);
  430. drm_crtc_send_vblank_event(crtc, event);
  431. spin_unlock_irq(&crtc->dev->event_lock);
  432. }
  433. }
  434. }
  435. static void armada_drm_crtc_atomic_enable(struct drm_crtc *crtc,
  436. struct drm_atomic_state *state)
  437. {
  438. struct drm_crtc_state *old_state = drm_atomic_get_old_crtc_state(state,
  439. crtc);
  440. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  441. DRM_DEBUG_KMS("[CRTC:%d:%s]\n", crtc->base.id, crtc->name);
  442. if (!old_state->active) {
  443. /*
  444. * This modeset is enabling the CRTC after it having
  445. * been disabled. Reverse the call to ->disable in
  446. * the atomic_disable().
  447. */
  448. if (dcrtc->variant->enable)
  449. dcrtc->variant->enable(dcrtc, &crtc->state->adjusted_mode);
  450. }
  451. armada_drm_crtc_update(dcrtc, true);
  452. drm_crtc_vblank_on(crtc);
  453. if (crtc->state->adjusted_mode.flags & DRM_MODE_FLAG_INTERLACE)
  454. WARN_ON(drm_crtc_vblank_get(crtc));
  455. armada_drm_crtc_queue_state_event(crtc);
  456. }
  457. static const struct drm_crtc_helper_funcs armada_crtc_helper_funcs = {
  458. .mode_valid = armada_drm_crtc_mode_valid,
  459. .mode_fixup = armada_drm_crtc_mode_fixup,
  460. .mode_set_nofb = armada_drm_crtc_mode_set_nofb,
  461. .atomic_check = armada_drm_crtc_atomic_check,
  462. .atomic_begin = armada_drm_crtc_atomic_begin,
  463. .atomic_flush = armada_drm_crtc_atomic_flush,
  464. .atomic_disable = armada_drm_crtc_atomic_disable,
  465. .atomic_enable = armada_drm_crtc_atomic_enable,
  466. };
  467. static void armada_load_cursor_argb(void __iomem *base, uint32_t *pix,
  468. unsigned stride, unsigned width, unsigned height)
  469. {
  470. uint32_t addr;
  471. unsigned y;
  472. addr = SRAM_HWC32_RAM1;
  473. for (y = 0; y < height; y++) {
  474. uint32_t *p = &pix[y * stride];
  475. unsigned x;
  476. for (x = 0; x < width; x++, p++) {
  477. uint32_t val = *p;
  478. /*
  479. * In "ARGB888" (HWC32) mode, writing to the SRAM
  480. * requires these bits to contain:
  481. * 31:24 = alpha 23:16 = blue 15:8 = green 7:0 = red
  482. * So, it's actually ABGR8888. This is independent
  483. * of the SWAPRB bits in DMA control register 0.
  484. */
  485. val = (val & 0xff00ff00) |
  486. (val & 0x000000ff) << 16 |
  487. (val & 0x00ff0000) >> 16;
  488. writel_relaxed(val,
  489. base + LCD_SPU_SRAM_WRDAT);
  490. writel_relaxed(addr | SRAM_WRITE,
  491. base + LCD_SPU_SRAM_CTRL);
  492. readl_relaxed(base + LCD_SPU_HWC_OVSA_HPXL_VLN);
  493. addr += 1;
  494. if ((addr & 0x00ff) == 0)
  495. addr += 0xf00;
  496. if ((addr & 0x30ff) == 0)
  497. addr = SRAM_HWC32_RAM2;
  498. }
  499. }
  500. }
  501. static void armada_drm_crtc_cursor_tran(void __iomem *base)
  502. {
  503. unsigned addr;
  504. for (addr = 0; addr < 256; addr++) {
  505. /* write the default value */
  506. writel_relaxed(0x55555555, base + LCD_SPU_SRAM_WRDAT);
  507. writel_relaxed(addr | SRAM_WRITE | SRAM_HWC32_TRAN,
  508. base + LCD_SPU_SRAM_CTRL);
  509. }
  510. }
  511. static int armada_drm_crtc_cursor_update(struct armada_crtc *dcrtc, bool reload)
  512. {
  513. uint32_t xoff, xscr, w = dcrtc->cursor_w, s;
  514. uint32_t yoff, yscr, h = dcrtc->cursor_h;
  515. uint32_t para1;
  516. /*
  517. * Calculate the visible width and height of the cursor,
  518. * screen position, and the position in the cursor bitmap.
  519. */
  520. if (dcrtc->cursor_x < 0) {
  521. xoff = -dcrtc->cursor_x;
  522. xscr = 0;
  523. w -= min(xoff, w);
  524. } else if (dcrtc->cursor_x + w > dcrtc->crtc.mode.hdisplay) {
  525. xoff = 0;
  526. xscr = dcrtc->cursor_x;
  527. w = max_t(int, dcrtc->crtc.mode.hdisplay - dcrtc->cursor_x, 0);
  528. } else {
  529. xoff = 0;
  530. xscr = dcrtc->cursor_x;
  531. }
  532. if (dcrtc->cursor_y < 0) {
  533. yoff = -dcrtc->cursor_y;
  534. yscr = 0;
  535. h -= min(yoff, h);
  536. } else if (dcrtc->cursor_y + h > dcrtc->crtc.mode.vdisplay) {
  537. yoff = 0;
  538. yscr = dcrtc->cursor_y;
  539. h = max_t(int, dcrtc->crtc.mode.vdisplay - dcrtc->cursor_y, 0);
  540. } else {
  541. yoff = 0;
  542. yscr = dcrtc->cursor_y;
  543. }
  544. /* On interlaced modes, the vertical cursor size must be halved */
  545. s = dcrtc->cursor_w;
  546. if (dcrtc->interlaced) {
  547. s *= 2;
  548. yscr /= 2;
  549. h /= 2;
  550. }
  551. if (!dcrtc->cursor_obj || !h || !w) {
  552. spin_lock_irq(&dcrtc->irq_lock);
  553. dcrtc->cursor_update = false;
  554. armada_updatel(0, CFG_HWC_ENA, dcrtc->base + LCD_SPU_DMA_CTRL0);
  555. spin_unlock_irq(&dcrtc->irq_lock);
  556. return 0;
  557. }
  558. spin_lock_irq(&dcrtc->irq_lock);
  559. para1 = readl_relaxed(dcrtc->base + LCD_SPU_SRAM_PARA1);
  560. armada_updatel(CFG_CSB_256x32, CFG_CSB_256x32 | CFG_PDWN256x32,
  561. dcrtc->base + LCD_SPU_SRAM_PARA1);
  562. spin_unlock_irq(&dcrtc->irq_lock);
  563. /*
  564. * Initialize the transparency if the SRAM was powered down.
  565. * We must also reload the cursor data as well.
  566. */
  567. if (!(para1 & CFG_CSB_256x32)) {
  568. armada_drm_crtc_cursor_tran(dcrtc->base);
  569. reload = true;
  570. }
  571. if (dcrtc->cursor_hw_sz != (h << 16 | w)) {
  572. spin_lock_irq(&dcrtc->irq_lock);
  573. dcrtc->cursor_update = false;
  574. armada_updatel(0, CFG_HWC_ENA, dcrtc->base + LCD_SPU_DMA_CTRL0);
  575. spin_unlock_irq(&dcrtc->irq_lock);
  576. reload = true;
  577. }
  578. if (reload) {
  579. struct armada_gem_object *obj = dcrtc->cursor_obj;
  580. uint32_t *pix;
  581. /* Set the top-left corner of the cursor image */
  582. pix = obj->addr;
  583. pix += yoff * s + xoff;
  584. armada_load_cursor_argb(dcrtc->base, pix, s, w, h);
  585. }
  586. /* Reload the cursor position, size and enable in the IRQ handler */
  587. spin_lock_irq(&dcrtc->irq_lock);
  588. dcrtc->cursor_hw_pos = yscr << 16 | xscr;
  589. dcrtc->cursor_hw_sz = h << 16 | w;
  590. dcrtc->cursor_update = true;
  591. armada_drm_crtc_enable_irq(dcrtc, DUMB_FRAMEDONE_ENA);
  592. spin_unlock_irq(&dcrtc->irq_lock);
  593. return 0;
  594. }
  595. static void cursor_update(void *data)
  596. {
  597. armada_drm_crtc_cursor_update(data, true);
  598. }
  599. static int armada_drm_crtc_cursor_set(struct drm_crtc *crtc,
  600. struct drm_file *file, uint32_t handle, uint32_t w, uint32_t h)
  601. {
  602. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  603. struct armada_gem_object *obj = NULL;
  604. int ret;
  605. /* If no cursor support, replicate drm's return value */
  606. if (!dcrtc->variant->has_spu_adv_reg)
  607. return -ENXIO;
  608. if (handle && w > 0 && h > 0) {
  609. /* maximum size is 64x32 or 32x64 */
  610. if (w > 64 || h > 64 || (w > 32 && h > 32))
  611. return -ENOMEM;
  612. obj = armada_gem_object_lookup(file, handle);
  613. if (!obj)
  614. return -ENOENT;
  615. /* Must be a kernel-mapped object */
  616. if (!obj->addr) {
  617. drm_gem_object_put(&obj->obj);
  618. return -EINVAL;
  619. }
  620. if (obj->obj.size < w * h * 4) {
  621. DRM_ERROR("buffer is too small\n");
  622. drm_gem_object_put(&obj->obj);
  623. return -ENOMEM;
  624. }
  625. }
  626. if (dcrtc->cursor_obj) {
  627. dcrtc->cursor_obj->update = NULL;
  628. dcrtc->cursor_obj->update_data = NULL;
  629. drm_gem_object_put(&dcrtc->cursor_obj->obj);
  630. }
  631. dcrtc->cursor_obj = obj;
  632. dcrtc->cursor_w = w;
  633. dcrtc->cursor_h = h;
  634. ret = armada_drm_crtc_cursor_update(dcrtc, true);
  635. if (obj) {
  636. obj->update_data = dcrtc;
  637. obj->update = cursor_update;
  638. }
  639. return ret;
  640. }
  641. static int armada_drm_crtc_cursor_move(struct drm_crtc *crtc, int x, int y)
  642. {
  643. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  644. int ret;
  645. /* If no cursor support, replicate drm's return value */
  646. if (!dcrtc->variant->has_spu_adv_reg)
  647. return -EFAULT;
  648. dcrtc->cursor_x = x;
  649. dcrtc->cursor_y = y;
  650. ret = armada_drm_crtc_cursor_update(dcrtc, false);
  651. return ret;
  652. }
  653. static void armada_drm_crtc_destroy(struct drm_crtc *crtc)
  654. {
  655. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  656. struct armada_private *priv = drm_to_armada_dev(crtc->dev);
  657. if (dcrtc->cursor_obj)
  658. drm_gem_object_put(&dcrtc->cursor_obj->obj);
  659. priv->dcrtc[dcrtc->num] = NULL;
  660. drm_crtc_cleanup(&dcrtc->crtc);
  661. if (dcrtc->variant->disable)
  662. dcrtc->variant->disable(dcrtc);
  663. writel_relaxed(0, dcrtc->base + LCD_SPU_IRQ_ENA);
  664. of_node_put(dcrtc->crtc.port);
  665. kfree(dcrtc);
  666. }
  667. static int armada_drm_crtc_late_register(struct drm_crtc *crtc)
  668. {
  669. if (IS_ENABLED(CONFIG_DEBUG_FS))
  670. armada_drm_crtc_debugfs_init(drm_to_armada_crtc(crtc));
  671. return 0;
  672. }
  673. /* These are called under the vbl_lock. */
  674. static int armada_drm_crtc_enable_vblank(struct drm_crtc *crtc)
  675. {
  676. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  677. unsigned long flags;
  678. spin_lock_irqsave(&dcrtc->irq_lock, flags);
  679. armada_drm_crtc_enable_irq(dcrtc, VSYNC_IRQ_ENA);
  680. spin_unlock_irqrestore(&dcrtc->irq_lock, flags);
  681. return 0;
  682. }
  683. static void armada_drm_crtc_disable_vblank(struct drm_crtc *crtc)
  684. {
  685. struct armada_crtc *dcrtc = drm_to_armada_crtc(crtc);
  686. unsigned long flags;
  687. spin_lock_irqsave(&dcrtc->irq_lock, flags);
  688. armada_drm_crtc_disable_irq(dcrtc, VSYNC_IRQ_ENA);
  689. spin_unlock_irqrestore(&dcrtc->irq_lock, flags);
  690. }
  691. static const struct drm_crtc_funcs armada_crtc_funcs = {
  692. .reset = drm_atomic_helper_crtc_reset,
  693. .cursor_set = armada_drm_crtc_cursor_set,
  694. .cursor_move = armada_drm_crtc_cursor_move,
  695. .destroy = armada_drm_crtc_destroy,
  696. .set_config = drm_atomic_helper_set_config,
  697. .page_flip = drm_atomic_helper_page_flip,
  698. .atomic_duplicate_state = drm_atomic_helper_crtc_duplicate_state,
  699. .atomic_destroy_state = drm_atomic_helper_crtc_destroy_state,
  700. .late_register = armada_drm_crtc_late_register,
  701. .enable_vblank = armada_drm_crtc_enable_vblank,
  702. .disable_vblank = armada_drm_crtc_disable_vblank,
  703. };
  704. int armada_crtc_select_clock(struct armada_crtc *dcrtc,
  705. struct armada_clk_result *res,
  706. const struct armada_clocking_params *params,
  707. struct clk *clks[], size_t num_clks,
  708. unsigned long desired_khz)
  709. {
  710. unsigned long desired_hz = desired_khz * 1000;
  711. unsigned long desired_clk_hz; // requested clk input
  712. unsigned long real_clk_hz; // actual clk input
  713. unsigned long real_hz; // actual pixel clk
  714. unsigned long permillage;
  715. struct clk *clk;
  716. u32 div;
  717. int i;
  718. DRM_DEBUG_KMS("[CRTC:%u:%s] desired clock=%luHz\n",
  719. dcrtc->crtc.base.id, dcrtc->crtc.name, desired_hz);
  720. for (i = 0; i < num_clks; i++) {
  721. clk = clks[i];
  722. if (!clk)
  723. continue;
  724. if (params->settable & BIT(i)) {
  725. real_clk_hz = clk_round_rate(clk, desired_hz);
  726. desired_clk_hz = desired_hz;
  727. } else {
  728. real_clk_hz = clk_get_rate(clk);
  729. desired_clk_hz = real_clk_hz;
  730. }
  731. /* If the clock can do exactly the desired rate, we're done */
  732. if (real_clk_hz == desired_hz) {
  733. real_hz = real_clk_hz;
  734. div = 1;
  735. goto found;
  736. }
  737. /* Calculate the divider - if invalid, we can't do this rate */
  738. div = DIV_ROUND_CLOSEST(real_clk_hz, desired_hz);
  739. if (div == 0 || div > params->div_max)
  740. continue;
  741. /* Calculate the actual rate - HDMI requires -0.6%..+0.5% */
  742. real_hz = DIV_ROUND_CLOSEST(real_clk_hz, div);
  743. DRM_DEBUG_KMS("[CRTC:%u:%s] clk=%u %luHz div=%u real=%luHz\n",
  744. dcrtc->crtc.base.id, dcrtc->crtc.name,
  745. i, real_clk_hz, div, real_hz);
  746. /* Avoid repeated division */
  747. if (real_hz < desired_hz) {
  748. permillage = real_hz / desired_khz;
  749. if (permillage < params->permillage_min)
  750. continue;
  751. } else {
  752. permillage = DIV_ROUND_UP(real_hz, desired_khz);
  753. if (permillage > params->permillage_max)
  754. continue;
  755. }
  756. goto found;
  757. }
  758. return -ERANGE;
  759. found:
  760. DRM_DEBUG_KMS("[CRTC:%u:%s] selected clk=%u %luHz div=%u real=%luHz\n",
  761. dcrtc->crtc.base.id, dcrtc->crtc.name,
  762. i, real_clk_hz, div, real_hz);
  763. res->desired_clk_hz = desired_clk_hz;
  764. res->clk = clk;
  765. res->div = div;
  766. return i;
  767. }
  768. static int armada_drm_crtc_create(struct drm_device *drm, struct device *dev,
  769. struct resource *res, int irq, const struct armada_variant *variant,
  770. struct device_node *port)
  771. {
  772. struct armada_private *priv = drm_to_armada_dev(drm);
  773. struct armada_crtc *dcrtc;
  774. struct drm_plane *primary;
  775. void __iomem *base;
  776. int ret;
  777. base = devm_ioremap_resource(dev, res);
  778. if (IS_ERR(base))
  779. return PTR_ERR(base);
  780. dcrtc = kzalloc(sizeof(*dcrtc), GFP_KERNEL);
  781. if (!dcrtc) {
  782. DRM_ERROR("failed to allocate Armada crtc\n");
  783. return -ENOMEM;
  784. }
  785. if (dev != drm->dev)
  786. dev_set_drvdata(dev, dcrtc);
  787. dcrtc->variant = variant;
  788. dcrtc->base = base;
  789. dcrtc->num = drm->mode_config.num_crtc;
  790. dcrtc->cfg_dumb_ctrl = DUMB24_RGB888_0;
  791. dcrtc->spu_iopad_ctrl = CFG_VSCALE_LN_EN | CFG_IOPAD_DUMB24;
  792. spin_lock_init(&dcrtc->irq_lock);
  793. dcrtc->irq_ena = CLEAN_SPU_IRQ_ISR;
  794. /* Initialize some registers which we don't otherwise set */
  795. writel_relaxed(0x00000001, dcrtc->base + LCD_CFG_SCLK_DIV);
  796. writel_relaxed(0x00000000, dcrtc->base + LCD_SPU_BLANKCOLOR);
  797. writel_relaxed(dcrtc->spu_iopad_ctrl,
  798. dcrtc->base + LCD_SPU_IOPAD_CONTROL);
  799. writel_relaxed(0x00000000, dcrtc->base + LCD_SPU_SRAM_PARA0);
  800. writel_relaxed(CFG_PDWN256x32 | CFG_PDWN256x24 | CFG_PDWN256x8 |
  801. CFG_PDWN32x32 | CFG_PDWN16x66 | CFG_PDWN32x66 |
  802. CFG_PDWN64x66, dcrtc->base + LCD_SPU_SRAM_PARA1);
  803. writel_relaxed(0x2032ff81, dcrtc->base + LCD_SPU_DMA_CTRL1);
  804. writel_relaxed(dcrtc->irq_ena, dcrtc->base + LCD_SPU_IRQ_ENA);
  805. readl_relaxed(dcrtc->base + LCD_SPU_IRQ_ISR);
  806. writel_relaxed(0, dcrtc->base + LCD_SPU_IRQ_ISR);
  807. ret = devm_request_irq(dev, irq, armada_drm_irq, 0, "armada_drm_crtc",
  808. dcrtc);
  809. if (ret < 0)
  810. goto err_crtc;
  811. if (dcrtc->variant->init) {
  812. ret = dcrtc->variant->init(dcrtc, dev);
  813. if (ret)
  814. goto err_crtc;
  815. }
  816. /* Ensure AXI pipeline is enabled */
  817. armada_updatel(CFG_ARBFAST_ENA, 0, dcrtc->base + LCD_SPU_DMA_CTRL0);
  818. priv->dcrtc[dcrtc->num] = dcrtc;
  819. dcrtc->crtc.port = port;
  820. primary = kzalloc(sizeof(*primary), GFP_KERNEL);
  821. if (!primary) {
  822. ret = -ENOMEM;
  823. goto err_crtc;
  824. }
  825. ret = armada_drm_primary_plane_init(drm, primary);
  826. if (ret) {
  827. kfree(primary);
  828. goto err_crtc;
  829. }
  830. ret = drm_crtc_init_with_planes(drm, &dcrtc->crtc, primary, NULL,
  831. &armada_crtc_funcs, NULL);
  832. if (ret)
  833. goto err_crtc_init;
  834. drm_crtc_helper_add(&dcrtc->crtc, &armada_crtc_helper_funcs);
  835. ret = drm_mode_crtc_set_gamma_size(&dcrtc->crtc, 256);
  836. if (ret)
  837. return ret;
  838. drm_crtc_enable_color_mgmt(&dcrtc->crtc, 0, false, 256);
  839. return armada_overlay_plane_create(drm, 1 << dcrtc->num);
  840. err_crtc_init:
  841. primary->funcs->destroy(primary);
  842. err_crtc:
  843. kfree(dcrtc);
  844. return ret;
  845. }
  846. static int
  847. armada_lcd_bind(struct device *dev, struct device *master, void *data)
  848. {
  849. struct platform_device *pdev = to_platform_device(dev);
  850. struct drm_device *drm = data;
  851. struct resource *res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  852. int irq = platform_get_irq(pdev, 0);
  853. const struct armada_variant *variant;
  854. struct device_node *port = NULL;
  855. if (irq < 0)
  856. return irq;
  857. if (!dev->of_node) {
  858. const struct platform_device_id *id;
  859. id = platform_get_device_id(pdev);
  860. if (!id)
  861. return -ENXIO;
  862. variant = (const struct armada_variant *)id->driver_data;
  863. } else {
  864. const struct of_device_id *match;
  865. struct device_node *np, *parent = dev->of_node;
  866. match = of_match_device(dev->driver->of_match_table, dev);
  867. if (!match)
  868. return -ENXIO;
  869. np = of_get_child_by_name(parent, "ports");
  870. if (np)
  871. parent = np;
  872. port = of_get_child_by_name(parent, "port");
  873. of_node_put(np);
  874. if (!port) {
  875. dev_err(dev, "no port node found in %pOF\n", parent);
  876. return -ENXIO;
  877. }
  878. variant = match->data;
  879. }
  880. return armada_drm_crtc_create(drm, dev, res, irq, variant, port);
  881. }
  882. static void
  883. armada_lcd_unbind(struct device *dev, struct device *master, void *data)
  884. {
  885. struct armada_crtc *dcrtc = dev_get_drvdata(dev);
  886. armada_drm_crtc_destroy(&dcrtc->crtc);
  887. }
  888. static const struct component_ops armada_lcd_ops = {
  889. .bind = armada_lcd_bind,
  890. .unbind = armada_lcd_unbind,
  891. };
  892. static int armada_lcd_probe(struct platform_device *pdev)
  893. {
  894. return component_add(&pdev->dev, &armada_lcd_ops);
  895. }
  896. static int armada_lcd_remove(struct platform_device *pdev)
  897. {
  898. component_del(&pdev->dev, &armada_lcd_ops);
  899. return 0;
  900. }
  901. static const struct of_device_id armada_lcd_of_match[] = {
  902. {
  903. .compatible = "marvell,dove-lcd",
  904. .data = &armada510_ops,
  905. },
  906. {}
  907. };
  908. MODULE_DEVICE_TABLE(of, armada_lcd_of_match);
  909. static const struct platform_device_id armada_lcd_platform_ids[] = {
  910. {
  911. .name = "armada-lcd",
  912. .driver_data = (unsigned long)&armada510_ops,
  913. }, {
  914. .name = "armada-510-lcd",
  915. .driver_data = (unsigned long)&armada510_ops,
  916. },
  917. { },
  918. };
  919. MODULE_DEVICE_TABLE(platform, armada_lcd_platform_ids);
  920. struct platform_driver armada_lcd_platform_driver = {
  921. .probe = armada_lcd_probe,
  922. .remove = armada_lcd_remove,
  923. .driver = {
  924. .name = "armada-lcd",
  925. .owner = THIS_MODULE,
  926. .of_match_table = armada_lcd_of_match,
  927. },
  928. .id_table = armada_lcd_platform_ids,
  929. };