sun4i_backend.c 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041
  1. // SPDX-License-Identifier: GPL-2.0-or-later
  2. /*
  3. * Copyright (C) 2015 Free Electrons
  4. * Copyright (C) 2015 NextThing Co
  5. *
  6. * Maxime Ripard <[email protected]>
  7. */
  8. #include <linux/component.h>
  9. #include <linux/list.h>
  10. #include <linux/module.h>
  11. #include <linux/of_device.h>
  12. #include <linux/of_graph.h>
  13. #include <linux/dma-mapping.h>
  14. #include <linux/platform_device.h>
  15. #include <linux/reset.h>
  16. #include <drm/drm_atomic.h>
  17. #include <drm/drm_atomic_helper.h>
  18. #include <drm/drm_blend.h>
  19. #include <drm/drm_crtc.h>
  20. #include <drm/drm_fb_dma_helper.h>
  21. #include <drm/drm_fourcc.h>
  22. #include <drm/drm_framebuffer.h>
  23. #include <drm/drm_gem_dma_helper.h>
  24. #include <drm/drm_probe_helper.h>
  25. #include "sun4i_backend.h"
  26. #include "sun4i_drv.h"
  27. #include "sun4i_frontend.h"
  28. #include "sun4i_layer.h"
  29. #include "sunxi_engine.h"
  30. struct sun4i_backend_quirks {
  31. /* backend <-> TCON muxing selection done in backend */
  32. bool needs_output_muxing;
  33. /* alpha at the lowest z position is not always supported */
  34. bool supports_lowest_plane_alpha;
  35. };
  36. static const u32 sunxi_rgb2yuv_coef[12] = {
  37. 0x00000107, 0x00000204, 0x00000064, 0x00000108,
  38. 0x00003f69, 0x00003ed6, 0x000001c1, 0x00000808,
  39. 0x000001c1, 0x00003e88, 0x00003fb8, 0x00000808
  40. };
  41. static void sun4i_backend_apply_color_correction(struct sunxi_engine *engine)
  42. {
  43. int i;
  44. DRM_DEBUG_DRIVER("Applying RGB to YUV color correction\n");
  45. /* Set color correction */
  46. regmap_write(engine->regs, SUN4I_BACKEND_OCCTL_REG,
  47. SUN4I_BACKEND_OCCTL_ENABLE);
  48. for (i = 0; i < 12; i++)
  49. regmap_write(engine->regs, SUN4I_BACKEND_OCRCOEF_REG(i),
  50. sunxi_rgb2yuv_coef[i]);
  51. }
  52. static void sun4i_backend_disable_color_correction(struct sunxi_engine *engine)
  53. {
  54. DRM_DEBUG_DRIVER("Disabling color correction\n");
  55. /* Disable color correction */
  56. regmap_update_bits(engine->regs, SUN4I_BACKEND_OCCTL_REG,
  57. SUN4I_BACKEND_OCCTL_ENABLE, 0);
  58. }
  59. static void sun4i_backend_commit(struct sunxi_engine *engine)
  60. {
  61. DRM_DEBUG_DRIVER("Committing changes\n");
  62. regmap_write(engine->regs, SUN4I_BACKEND_REGBUFFCTL_REG,
  63. SUN4I_BACKEND_REGBUFFCTL_AUTOLOAD_DIS |
  64. SUN4I_BACKEND_REGBUFFCTL_LOADCTL);
  65. }
  66. void sun4i_backend_layer_enable(struct sun4i_backend *backend,
  67. int layer, bool enable)
  68. {
  69. u32 val;
  70. DRM_DEBUG_DRIVER("%sabling layer %d\n", enable ? "En" : "Dis",
  71. layer);
  72. if (enable)
  73. val = SUN4I_BACKEND_MODCTL_LAY_EN(layer);
  74. else
  75. val = 0;
  76. regmap_update_bits(backend->engine.regs, SUN4I_BACKEND_MODCTL_REG,
  77. SUN4I_BACKEND_MODCTL_LAY_EN(layer), val);
  78. }
  79. static int sun4i_backend_drm_format_to_layer(u32 format, u32 *mode)
  80. {
  81. switch (format) {
  82. case DRM_FORMAT_ARGB8888:
  83. *mode = SUN4I_BACKEND_LAY_FBFMT_ARGB8888;
  84. break;
  85. case DRM_FORMAT_ARGB4444:
  86. *mode = SUN4I_BACKEND_LAY_FBFMT_ARGB4444;
  87. break;
  88. case DRM_FORMAT_ARGB1555:
  89. *mode = SUN4I_BACKEND_LAY_FBFMT_ARGB1555;
  90. break;
  91. case DRM_FORMAT_RGBA5551:
  92. *mode = SUN4I_BACKEND_LAY_FBFMT_RGBA5551;
  93. break;
  94. case DRM_FORMAT_RGBA4444:
  95. *mode = SUN4I_BACKEND_LAY_FBFMT_RGBA4444;
  96. break;
  97. case DRM_FORMAT_XRGB8888:
  98. *mode = SUN4I_BACKEND_LAY_FBFMT_XRGB8888;
  99. break;
  100. case DRM_FORMAT_RGB888:
  101. *mode = SUN4I_BACKEND_LAY_FBFMT_RGB888;
  102. break;
  103. case DRM_FORMAT_RGB565:
  104. *mode = SUN4I_BACKEND_LAY_FBFMT_RGB565;
  105. break;
  106. default:
  107. return -EINVAL;
  108. }
  109. return 0;
  110. }
  111. static const uint32_t sun4i_backend_formats[] = {
  112. DRM_FORMAT_ARGB1555,
  113. DRM_FORMAT_ARGB4444,
  114. DRM_FORMAT_ARGB8888,
  115. DRM_FORMAT_RGB565,
  116. DRM_FORMAT_RGB888,
  117. DRM_FORMAT_RGBA4444,
  118. DRM_FORMAT_RGBA5551,
  119. DRM_FORMAT_UYVY,
  120. DRM_FORMAT_VYUY,
  121. DRM_FORMAT_XRGB8888,
  122. DRM_FORMAT_YUYV,
  123. DRM_FORMAT_YVYU,
  124. };
  125. bool sun4i_backend_format_is_supported(uint32_t fmt, uint64_t modifier)
  126. {
  127. unsigned int i;
  128. if (modifier != DRM_FORMAT_MOD_LINEAR)
  129. return false;
  130. for (i = 0; i < ARRAY_SIZE(sun4i_backend_formats); i++)
  131. if (sun4i_backend_formats[i] == fmt)
  132. return true;
  133. return false;
  134. }
  135. int sun4i_backend_update_layer_coord(struct sun4i_backend *backend,
  136. int layer, struct drm_plane *plane)
  137. {
  138. struct drm_plane_state *state = plane->state;
  139. DRM_DEBUG_DRIVER("Updating layer %d\n", layer);
  140. /* Set height and width */
  141. DRM_DEBUG_DRIVER("Layer size W: %u H: %u\n",
  142. state->crtc_w, state->crtc_h);
  143. regmap_write(backend->engine.regs, SUN4I_BACKEND_LAYSIZE_REG(layer),
  144. SUN4I_BACKEND_LAYSIZE(state->crtc_w,
  145. state->crtc_h));
  146. /* Set base coordinates */
  147. DRM_DEBUG_DRIVER("Layer coordinates X: %d Y: %d\n",
  148. state->crtc_x, state->crtc_y);
  149. regmap_write(backend->engine.regs, SUN4I_BACKEND_LAYCOOR_REG(layer),
  150. SUN4I_BACKEND_LAYCOOR(state->crtc_x,
  151. state->crtc_y));
  152. return 0;
  153. }
  154. static int sun4i_backend_update_yuv_format(struct sun4i_backend *backend,
  155. int layer, struct drm_plane *plane)
  156. {
  157. struct drm_plane_state *state = plane->state;
  158. struct drm_framebuffer *fb = state->fb;
  159. const struct drm_format_info *format = fb->format;
  160. const uint32_t fmt = format->format;
  161. u32 val = SUN4I_BACKEND_IYUVCTL_EN;
  162. int i;
  163. for (i = 0; i < ARRAY_SIZE(sunxi_bt601_yuv2rgb_coef); i++)
  164. regmap_write(backend->engine.regs,
  165. SUN4I_BACKEND_YGCOEF_REG(i),
  166. sunxi_bt601_yuv2rgb_coef[i]);
  167. /*
  168. * We should do that only for a single plane, but the
  169. * framebuffer's atomic_check has our back on this.
  170. */
  171. regmap_update_bits(backend->engine.regs, SUN4I_BACKEND_ATTCTL_REG0(layer),
  172. SUN4I_BACKEND_ATTCTL_REG0_LAY_YUVEN,
  173. SUN4I_BACKEND_ATTCTL_REG0_LAY_YUVEN);
  174. /* TODO: Add support for the multi-planar YUV formats */
  175. if (drm_format_info_is_yuv_packed(format) &&
  176. drm_format_info_is_yuv_sampling_422(format))
  177. val |= SUN4I_BACKEND_IYUVCTL_FBFMT_PACKED_YUV422;
  178. else
  179. DRM_DEBUG_DRIVER("Unsupported YUV format (0x%x)\n", fmt);
  180. /*
  181. * Allwinner seems to list the pixel sequence from right to left, while
  182. * DRM lists it from left to right.
  183. */
  184. switch (fmt) {
  185. case DRM_FORMAT_YUYV:
  186. val |= SUN4I_BACKEND_IYUVCTL_FBPS_VYUY;
  187. break;
  188. case DRM_FORMAT_YVYU:
  189. val |= SUN4I_BACKEND_IYUVCTL_FBPS_UYVY;
  190. break;
  191. case DRM_FORMAT_UYVY:
  192. val |= SUN4I_BACKEND_IYUVCTL_FBPS_YVYU;
  193. break;
  194. case DRM_FORMAT_VYUY:
  195. val |= SUN4I_BACKEND_IYUVCTL_FBPS_YUYV;
  196. break;
  197. default:
  198. DRM_DEBUG_DRIVER("Unsupported YUV pixel sequence (0x%x)\n",
  199. fmt);
  200. }
  201. regmap_write(backend->engine.regs, SUN4I_BACKEND_IYUVCTL_REG, val);
  202. return 0;
  203. }
  204. int sun4i_backend_update_layer_formats(struct sun4i_backend *backend,
  205. int layer, struct drm_plane *plane)
  206. {
  207. struct drm_plane_state *state = plane->state;
  208. struct drm_framebuffer *fb = state->fb;
  209. u32 val;
  210. int ret;
  211. /* Clear the YUV mode */
  212. regmap_update_bits(backend->engine.regs, SUN4I_BACKEND_ATTCTL_REG0(layer),
  213. SUN4I_BACKEND_ATTCTL_REG0_LAY_YUVEN, 0);
  214. val = SUN4I_BACKEND_ATTCTL_REG0_LAY_GLBALPHA(state->alpha >> 8);
  215. if (state->alpha != DRM_BLEND_ALPHA_OPAQUE)
  216. val |= SUN4I_BACKEND_ATTCTL_REG0_LAY_GLBALPHA_EN;
  217. regmap_update_bits(backend->engine.regs,
  218. SUN4I_BACKEND_ATTCTL_REG0(layer),
  219. SUN4I_BACKEND_ATTCTL_REG0_LAY_GLBALPHA_MASK |
  220. SUN4I_BACKEND_ATTCTL_REG0_LAY_GLBALPHA_EN,
  221. val);
  222. if (fb->format->is_yuv)
  223. return sun4i_backend_update_yuv_format(backend, layer, plane);
  224. ret = sun4i_backend_drm_format_to_layer(fb->format->format, &val);
  225. if (ret) {
  226. DRM_DEBUG_DRIVER("Invalid format\n");
  227. return ret;
  228. }
  229. regmap_update_bits(backend->engine.regs,
  230. SUN4I_BACKEND_ATTCTL_REG1(layer),
  231. SUN4I_BACKEND_ATTCTL_REG1_LAY_FBFMT, val);
  232. return 0;
  233. }
  234. int sun4i_backend_update_layer_frontend(struct sun4i_backend *backend,
  235. int layer, uint32_t fmt)
  236. {
  237. u32 val;
  238. int ret;
  239. ret = sun4i_backend_drm_format_to_layer(fmt, &val);
  240. if (ret) {
  241. DRM_DEBUG_DRIVER("Invalid format\n");
  242. return ret;
  243. }
  244. regmap_update_bits(backend->engine.regs,
  245. SUN4I_BACKEND_ATTCTL_REG0(layer),
  246. SUN4I_BACKEND_ATTCTL_REG0_LAY_VDOEN,
  247. SUN4I_BACKEND_ATTCTL_REG0_LAY_VDOEN);
  248. regmap_update_bits(backend->engine.regs,
  249. SUN4I_BACKEND_ATTCTL_REG1(layer),
  250. SUN4I_BACKEND_ATTCTL_REG1_LAY_FBFMT, val);
  251. return 0;
  252. }
  253. static int sun4i_backend_update_yuv_buffer(struct sun4i_backend *backend,
  254. struct drm_framebuffer *fb,
  255. dma_addr_t paddr)
  256. {
  257. /* TODO: Add support for the multi-planar YUV formats */
  258. DRM_DEBUG_DRIVER("Setting packed YUV buffer address to %pad\n", &paddr);
  259. regmap_write(backend->engine.regs, SUN4I_BACKEND_IYUVADD_REG(0), paddr);
  260. DRM_DEBUG_DRIVER("Layer line width: %d bits\n", fb->pitches[0] * 8);
  261. regmap_write(backend->engine.regs, SUN4I_BACKEND_IYUVLINEWIDTH_REG(0),
  262. fb->pitches[0] * 8);
  263. return 0;
  264. }
  265. int sun4i_backend_update_layer_buffer(struct sun4i_backend *backend,
  266. int layer, struct drm_plane *plane)
  267. {
  268. struct drm_plane_state *state = plane->state;
  269. struct drm_framebuffer *fb = state->fb;
  270. u32 lo_paddr, hi_paddr;
  271. dma_addr_t dma_addr;
  272. /* Set the line width */
  273. DRM_DEBUG_DRIVER("Layer line width: %d bits\n", fb->pitches[0] * 8);
  274. regmap_write(backend->engine.regs,
  275. SUN4I_BACKEND_LAYLINEWIDTH_REG(layer),
  276. fb->pitches[0] * 8);
  277. /* Get the start of the displayed memory */
  278. dma_addr = drm_fb_dma_get_gem_addr(fb, state, 0);
  279. DRM_DEBUG_DRIVER("Setting buffer address to %pad\n", &dma_addr);
  280. if (fb->format->is_yuv)
  281. return sun4i_backend_update_yuv_buffer(backend, fb, dma_addr);
  282. /* Write the 32 lower bits of the address (in bits) */
  283. lo_paddr = dma_addr << 3;
  284. DRM_DEBUG_DRIVER("Setting address lower bits to 0x%x\n", lo_paddr);
  285. regmap_write(backend->engine.regs,
  286. SUN4I_BACKEND_LAYFB_L32ADD_REG(layer),
  287. lo_paddr);
  288. /* And the upper bits */
  289. hi_paddr = dma_addr >> 29;
  290. DRM_DEBUG_DRIVER("Setting address high bits to 0x%x\n", hi_paddr);
  291. regmap_update_bits(backend->engine.regs, SUN4I_BACKEND_LAYFB_H4ADD_REG,
  292. SUN4I_BACKEND_LAYFB_H4ADD_MSK(layer),
  293. SUN4I_BACKEND_LAYFB_H4ADD(layer, hi_paddr));
  294. return 0;
  295. }
  296. int sun4i_backend_update_layer_zpos(struct sun4i_backend *backend, int layer,
  297. struct drm_plane *plane)
  298. {
  299. struct drm_plane_state *state = plane->state;
  300. struct sun4i_layer_state *p_state = state_to_sun4i_layer_state(state);
  301. unsigned int priority = state->normalized_zpos;
  302. unsigned int pipe = p_state->pipe;
  303. DRM_DEBUG_DRIVER("Setting layer %d's priority to %d and pipe %d\n",
  304. layer, priority, pipe);
  305. regmap_update_bits(backend->engine.regs, SUN4I_BACKEND_ATTCTL_REG0(layer),
  306. SUN4I_BACKEND_ATTCTL_REG0_LAY_PIPESEL_MASK |
  307. SUN4I_BACKEND_ATTCTL_REG0_LAY_PRISEL_MASK,
  308. SUN4I_BACKEND_ATTCTL_REG0_LAY_PIPESEL(p_state->pipe) |
  309. SUN4I_BACKEND_ATTCTL_REG0_LAY_PRISEL(priority));
  310. return 0;
  311. }
  312. void sun4i_backend_cleanup_layer(struct sun4i_backend *backend,
  313. int layer)
  314. {
  315. regmap_update_bits(backend->engine.regs,
  316. SUN4I_BACKEND_ATTCTL_REG0(layer),
  317. SUN4I_BACKEND_ATTCTL_REG0_LAY_VDOEN |
  318. SUN4I_BACKEND_ATTCTL_REG0_LAY_YUVEN, 0);
  319. }
  320. static bool sun4i_backend_plane_uses_scaler(struct drm_plane_state *state)
  321. {
  322. u16 src_h = state->src_h >> 16;
  323. u16 src_w = state->src_w >> 16;
  324. DRM_DEBUG_DRIVER("Input size %dx%d, output size %dx%d\n",
  325. src_w, src_h, state->crtc_w, state->crtc_h);
  326. if ((state->crtc_h != src_h) || (state->crtc_w != src_w))
  327. return true;
  328. return false;
  329. }
  330. static bool sun4i_backend_plane_uses_frontend(struct drm_plane_state *state)
  331. {
  332. struct sun4i_layer *layer = plane_to_sun4i_layer(state->plane);
  333. struct sun4i_backend *backend = layer->backend;
  334. uint32_t format = state->fb->format->format;
  335. uint64_t modifier = state->fb->modifier;
  336. if (IS_ERR(backend->frontend))
  337. return false;
  338. if (!sun4i_frontend_format_is_supported(format, modifier))
  339. return false;
  340. if (!sun4i_backend_format_is_supported(format, modifier))
  341. return true;
  342. /*
  343. * TODO: The backend alone allows 2x and 4x integer scaling, including
  344. * support for an alpha component (which the frontend doesn't support).
  345. * Use the backend directly instead of the frontend in this case, with
  346. * another test to return false.
  347. */
  348. if (sun4i_backend_plane_uses_scaler(state))
  349. return true;
  350. /*
  351. * Here the format is supported by both the frontend and the backend
  352. * and no frontend scaling is required, so use the backend directly.
  353. */
  354. return false;
  355. }
  356. static bool sun4i_backend_plane_is_supported(struct drm_plane_state *state,
  357. bool *uses_frontend)
  358. {
  359. if (sun4i_backend_plane_uses_frontend(state)) {
  360. *uses_frontend = true;
  361. return true;
  362. }
  363. *uses_frontend = false;
  364. /* Scaling is not supported without the frontend. */
  365. if (sun4i_backend_plane_uses_scaler(state))
  366. return false;
  367. return true;
  368. }
  369. static void sun4i_backend_atomic_begin(struct sunxi_engine *engine,
  370. struct drm_crtc_state *old_state)
  371. {
  372. u32 val;
  373. WARN_ON(regmap_read_poll_timeout(engine->regs,
  374. SUN4I_BACKEND_REGBUFFCTL_REG,
  375. val, !(val & SUN4I_BACKEND_REGBUFFCTL_LOADCTL),
  376. 100, 50000));
  377. }
  378. static int sun4i_backend_atomic_check(struct sunxi_engine *engine,
  379. struct drm_crtc_state *crtc_state)
  380. {
  381. struct drm_plane_state *plane_states[SUN4I_BACKEND_NUM_LAYERS] = { 0 };
  382. struct sun4i_backend *backend = engine_to_sun4i_backend(engine);
  383. struct drm_atomic_state *state = crtc_state->state;
  384. struct drm_device *drm = state->dev;
  385. struct drm_plane *plane;
  386. unsigned int num_planes = 0;
  387. unsigned int num_alpha_planes = 0;
  388. unsigned int num_frontend_planes = 0;
  389. unsigned int num_alpha_planes_max = 1;
  390. unsigned int num_yuv_planes = 0;
  391. unsigned int current_pipe = 0;
  392. unsigned int i;
  393. DRM_DEBUG_DRIVER("Starting checking our planes\n");
  394. if (!crtc_state->planes_changed)
  395. return 0;
  396. drm_for_each_plane_mask(plane, drm, crtc_state->plane_mask) {
  397. struct drm_plane_state *plane_state =
  398. drm_atomic_get_plane_state(state, plane);
  399. struct sun4i_layer_state *layer_state =
  400. state_to_sun4i_layer_state(plane_state);
  401. struct drm_framebuffer *fb = plane_state->fb;
  402. if (!sun4i_backend_plane_is_supported(plane_state,
  403. &layer_state->uses_frontend))
  404. return -EINVAL;
  405. if (layer_state->uses_frontend) {
  406. DRM_DEBUG_DRIVER("Using the frontend for plane %d\n",
  407. plane->index);
  408. num_frontend_planes++;
  409. } else {
  410. if (fb->format->is_yuv) {
  411. DRM_DEBUG_DRIVER("Plane FB format is YUV\n");
  412. num_yuv_planes++;
  413. }
  414. }
  415. DRM_DEBUG_DRIVER("Plane FB format is %p4cc\n",
  416. &fb->format->format);
  417. if (fb->format->has_alpha || (plane_state->alpha != DRM_BLEND_ALPHA_OPAQUE))
  418. num_alpha_planes++;
  419. DRM_DEBUG_DRIVER("Plane zpos is %d\n",
  420. plane_state->normalized_zpos);
  421. /* Sort our planes by Zpos */
  422. plane_states[plane_state->normalized_zpos] = plane_state;
  423. num_planes++;
  424. }
  425. /* All our planes were disabled, bail out */
  426. if (!num_planes)
  427. return 0;
  428. /*
  429. * The hardware is a bit unusual here.
  430. *
  431. * Even though it supports 4 layers, it does the composition
  432. * in two separate steps.
  433. *
  434. * The first one is assigning a layer to one of its two
  435. * pipes. If more that 1 layer is assigned to the same pipe,
  436. * and if pixels overlaps, the pipe will take the pixel from
  437. * the layer with the highest priority.
  438. *
  439. * The second step is the actual alpha blending, that takes
  440. * the two pipes as input, and uses the potential alpha
  441. * component to do the transparency between the two.
  442. *
  443. * This two-step scenario makes us unable to guarantee a
  444. * robust alpha blending between the 4 layers in all
  445. * situations, since this means that we need to have one layer
  446. * with alpha at the lowest position of our two pipes.
  447. *
  448. * However, we cannot even do that on every platform, since
  449. * the hardware has a bug where the lowest plane of the lowest
  450. * pipe (pipe 0, priority 0), if it has any alpha, will
  451. * discard the pixel data entirely and just display the pixels
  452. * in the background color (black by default).
  453. *
  454. * This means that on the affected platforms, we effectively
  455. * have only three valid configurations with alpha, all of
  456. * them with the alpha being on pipe1 with the lowest
  457. * position, which can be 1, 2 or 3 depending on the number of
  458. * planes and their zpos.
  459. */
  460. /* For platforms that are not affected by the issue described above. */
  461. if (backend->quirks->supports_lowest_plane_alpha)
  462. num_alpha_planes_max++;
  463. if (num_alpha_planes > num_alpha_planes_max) {
  464. DRM_DEBUG_DRIVER("Too many planes with alpha, rejecting...\n");
  465. return -EINVAL;
  466. }
  467. /* We can't have an alpha plane at the lowest position */
  468. if (!backend->quirks->supports_lowest_plane_alpha &&
  469. (plane_states[0]->alpha != DRM_BLEND_ALPHA_OPAQUE))
  470. return -EINVAL;
  471. for (i = 1; i < num_planes; i++) {
  472. struct drm_plane_state *p_state = plane_states[i];
  473. struct drm_framebuffer *fb = p_state->fb;
  474. struct sun4i_layer_state *s_state = state_to_sun4i_layer_state(p_state);
  475. /*
  476. * The only alpha position is the lowest plane of the
  477. * second pipe.
  478. */
  479. if (fb->format->has_alpha || (p_state->alpha != DRM_BLEND_ALPHA_OPAQUE))
  480. current_pipe++;
  481. s_state->pipe = current_pipe;
  482. }
  483. /* We can only have a single YUV plane at a time */
  484. if (num_yuv_planes > SUN4I_BACKEND_NUM_YUV_PLANES) {
  485. DRM_DEBUG_DRIVER("Too many planes with YUV, rejecting...\n");
  486. return -EINVAL;
  487. }
  488. if (num_frontend_planes > SUN4I_BACKEND_NUM_FRONTEND_LAYERS) {
  489. DRM_DEBUG_DRIVER("Too many planes going through the frontend, rejecting\n");
  490. return -EINVAL;
  491. }
  492. DRM_DEBUG_DRIVER("State valid with %u planes, %u alpha, %u video, %u YUV\n",
  493. num_planes, num_alpha_planes, num_frontend_planes,
  494. num_yuv_planes);
  495. return 0;
  496. }
  497. static void sun4i_backend_vblank_quirk(struct sunxi_engine *engine)
  498. {
  499. struct sun4i_backend *backend = engine_to_sun4i_backend(engine);
  500. struct sun4i_frontend *frontend = backend->frontend;
  501. if (!frontend)
  502. return;
  503. /*
  504. * In a teardown scenario with the frontend involved, we have
  505. * to keep the frontend enabled until the next vblank, and
  506. * only then disable it.
  507. *
  508. * This is due to the fact that the backend will not take into
  509. * account the new configuration (with the plane that used to
  510. * be fed by the frontend now disabled) until we write to the
  511. * commit bit and the hardware fetches the new configuration
  512. * during the next vblank.
  513. *
  514. * So we keep the frontend around in order to prevent any
  515. * visual artifacts.
  516. */
  517. spin_lock(&backend->frontend_lock);
  518. if (backend->frontend_teardown) {
  519. sun4i_frontend_exit(frontend);
  520. backend->frontend_teardown = false;
  521. }
  522. spin_unlock(&backend->frontend_lock);
  523. };
  524. static void sun4i_backend_mode_set(struct sunxi_engine *engine,
  525. const struct drm_display_mode *mode)
  526. {
  527. bool interlaced = !!(mode->flags & DRM_MODE_FLAG_INTERLACE);
  528. DRM_DEBUG_DRIVER("Updating global size W: %u H: %u\n",
  529. mode->hdisplay, mode->vdisplay);
  530. regmap_write(engine->regs, SUN4I_BACKEND_DISSIZE_REG,
  531. SUN4I_BACKEND_DISSIZE(mode->hdisplay, mode->vdisplay));
  532. regmap_update_bits(engine->regs, SUN4I_BACKEND_MODCTL_REG,
  533. SUN4I_BACKEND_MODCTL_ITLMOD_EN,
  534. interlaced ? SUN4I_BACKEND_MODCTL_ITLMOD_EN : 0);
  535. DRM_DEBUG_DRIVER("Switching display backend interlaced mode %s\n",
  536. interlaced ? "on" : "off");
  537. }
  538. static int sun4i_backend_init_sat(struct device *dev) {
  539. struct sun4i_backend *backend = dev_get_drvdata(dev);
  540. int ret;
  541. backend->sat_reset = devm_reset_control_get(dev, "sat");
  542. if (IS_ERR(backend->sat_reset)) {
  543. dev_err(dev, "Couldn't get the SAT reset line\n");
  544. return PTR_ERR(backend->sat_reset);
  545. }
  546. ret = reset_control_deassert(backend->sat_reset);
  547. if (ret) {
  548. dev_err(dev, "Couldn't deassert the SAT reset line\n");
  549. return ret;
  550. }
  551. backend->sat_clk = devm_clk_get(dev, "sat");
  552. if (IS_ERR(backend->sat_clk)) {
  553. dev_err(dev, "Couldn't get our SAT clock\n");
  554. ret = PTR_ERR(backend->sat_clk);
  555. goto err_assert_reset;
  556. }
  557. ret = clk_prepare_enable(backend->sat_clk);
  558. if (ret) {
  559. dev_err(dev, "Couldn't enable the SAT clock\n");
  560. return ret;
  561. }
  562. return 0;
  563. err_assert_reset:
  564. reset_control_assert(backend->sat_reset);
  565. return ret;
  566. }
  567. static int sun4i_backend_free_sat(struct device *dev) {
  568. struct sun4i_backend *backend = dev_get_drvdata(dev);
  569. clk_disable_unprepare(backend->sat_clk);
  570. reset_control_assert(backend->sat_reset);
  571. return 0;
  572. }
  573. /*
  574. * The display backend can take video output from the display frontend, or
  575. * the display enhancement unit on the A80, as input for one it its layers.
  576. * This relationship within the display pipeline is encoded in the device
  577. * tree with of_graph, and we use it here to figure out which backend, if
  578. * there are 2 or more, we are currently probing. The number would be in
  579. * the "reg" property of the upstream output port endpoint.
  580. */
  581. static int sun4i_backend_of_get_id(struct device_node *node)
  582. {
  583. struct device_node *ep, *remote;
  584. struct of_endpoint of_ep;
  585. /* Input port is 0, and we want the first endpoint. */
  586. ep = of_graph_get_endpoint_by_regs(node, 0, -1);
  587. if (!ep)
  588. return -EINVAL;
  589. remote = of_graph_get_remote_endpoint(ep);
  590. of_node_put(ep);
  591. if (!remote)
  592. return -EINVAL;
  593. of_graph_parse_endpoint(remote, &of_ep);
  594. of_node_put(remote);
  595. return of_ep.id;
  596. }
  597. /* TODO: This needs to take multiple pipelines into account */
  598. static struct sun4i_frontend *sun4i_backend_find_frontend(struct sun4i_drv *drv,
  599. struct device_node *node)
  600. {
  601. struct device_node *port, *ep, *remote;
  602. struct sun4i_frontend *frontend;
  603. port = of_graph_get_port_by_id(node, 0);
  604. if (!port)
  605. return ERR_PTR(-EINVAL);
  606. for_each_available_child_of_node(port, ep) {
  607. remote = of_graph_get_remote_port_parent(ep);
  608. if (!remote)
  609. continue;
  610. of_node_put(remote);
  611. /* does this node match any registered engines? */
  612. list_for_each_entry(frontend, &drv->frontend_list, list) {
  613. if (remote == frontend->node) {
  614. of_node_put(port);
  615. of_node_put(ep);
  616. return frontend;
  617. }
  618. }
  619. }
  620. of_node_put(port);
  621. return ERR_PTR(-EINVAL);
  622. }
  623. static const struct sunxi_engine_ops sun4i_backend_engine_ops = {
  624. .atomic_begin = sun4i_backend_atomic_begin,
  625. .atomic_check = sun4i_backend_atomic_check,
  626. .commit = sun4i_backend_commit,
  627. .layers_init = sun4i_layers_init,
  628. .apply_color_correction = sun4i_backend_apply_color_correction,
  629. .disable_color_correction = sun4i_backend_disable_color_correction,
  630. .vblank_quirk = sun4i_backend_vblank_quirk,
  631. .mode_set = sun4i_backend_mode_set,
  632. };
  633. static const struct regmap_config sun4i_backend_regmap_config = {
  634. .reg_bits = 32,
  635. .val_bits = 32,
  636. .reg_stride = 4,
  637. .max_register = 0x5800,
  638. };
  639. static int sun4i_backend_bind(struct device *dev, struct device *master,
  640. void *data)
  641. {
  642. struct platform_device *pdev = to_platform_device(dev);
  643. struct drm_device *drm = data;
  644. struct sun4i_drv *drv = drm->dev_private;
  645. struct sun4i_backend *backend;
  646. const struct sun4i_backend_quirks *quirks;
  647. void __iomem *regs;
  648. int i, ret;
  649. backend = devm_kzalloc(dev, sizeof(*backend), GFP_KERNEL);
  650. if (!backend)
  651. return -ENOMEM;
  652. dev_set_drvdata(dev, backend);
  653. spin_lock_init(&backend->frontend_lock);
  654. if (of_find_property(dev->of_node, "interconnects", NULL)) {
  655. /*
  656. * This assume we have the same DMA constraints for all our the
  657. * devices in our pipeline (all the backends, but also the
  658. * frontends). This sounds bad, but it has always been the case
  659. * for us, and DRM doesn't do per-device allocation either, so
  660. * we would need to fix DRM first...
  661. */
  662. ret = of_dma_configure(drm->dev, dev->of_node, true);
  663. if (ret)
  664. return ret;
  665. }
  666. backend->engine.node = dev->of_node;
  667. backend->engine.ops = &sun4i_backend_engine_ops;
  668. backend->engine.id = sun4i_backend_of_get_id(dev->of_node);
  669. if (backend->engine.id < 0)
  670. return backend->engine.id;
  671. backend->frontend = sun4i_backend_find_frontend(drv, dev->of_node);
  672. if (IS_ERR(backend->frontend))
  673. dev_warn(dev, "Couldn't find matching frontend, frontend features disabled\n");
  674. regs = devm_platform_ioremap_resource(pdev, 0);
  675. if (IS_ERR(regs))
  676. return PTR_ERR(regs);
  677. backend->reset = devm_reset_control_get(dev, NULL);
  678. if (IS_ERR(backend->reset)) {
  679. dev_err(dev, "Couldn't get our reset line\n");
  680. return PTR_ERR(backend->reset);
  681. }
  682. ret = reset_control_deassert(backend->reset);
  683. if (ret) {
  684. dev_err(dev, "Couldn't deassert our reset line\n");
  685. return ret;
  686. }
  687. backend->bus_clk = devm_clk_get(dev, "ahb");
  688. if (IS_ERR(backend->bus_clk)) {
  689. dev_err(dev, "Couldn't get the backend bus clock\n");
  690. ret = PTR_ERR(backend->bus_clk);
  691. goto err_assert_reset;
  692. }
  693. clk_prepare_enable(backend->bus_clk);
  694. backend->mod_clk = devm_clk_get(dev, "mod");
  695. if (IS_ERR(backend->mod_clk)) {
  696. dev_err(dev, "Couldn't get the backend module clock\n");
  697. ret = PTR_ERR(backend->mod_clk);
  698. goto err_disable_bus_clk;
  699. }
  700. ret = clk_set_rate_exclusive(backend->mod_clk, 300000000);
  701. if (ret) {
  702. dev_err(dev, "Couldn't set the module clock frequency\n");
  703. goto err_disable_bus_clk;
  704. }
  705. clk_prepare_enable(backend->mod_clk);
  706. backend->ram_clk = devm_clk_get(dev, "ram");
  707. if (IS_ERR(backend->ram_clk)) {
  708. dev_err(dev, "Couldn't get the backend RAM clock\n");
  709. ret = PTR_ERR(backend->ram_clk);
  710. goto err_disable_mod_clk;
  711. }
  712. clk_prepare_enable(backend->ram_clk);
  713. if (of_device_is_compatible(dev->of_node,
  714. "allwinner,sun8i-a33-display-backend")) {
  715. ret = sun4i_backend_init_sat(dev);
  716. if (ret) {
  717. dev_err(dev, "Couldn't init SAT resources\n");
  718. goto err_disable_ram_clk;
  719. }
  720. }
  721. backend->engine.regs = devm_regmap_init_mmio(dev, regs,
  722. &sun4i_backend_regmap_config);
  723. if (IS_ERR(backend->engine.regs)) {
  724. dev_err(dev, "Couldn't create the backend regmap\n");
  725. return PTR_ERR(backend->engine.regs);
  726. }
  727. list_add_tail(&backend->engine.list, &drv->engine_list);
  728. /*
  729. * Many of the backend's layer configuration registers have
  730. * undefined default values. This poses a risk as we use
  731. * regmap_update_bits in some places, and don't overwrite
  732. * the whole register.
  733. *
  734. * Clear the registers here to have something predictable.
  735. */
  736. for (i = 0x800; i < 0x1000; i += 4)
  737. regmap_write(backend->engine.regs, i, 0);
  738. /* Disable registers autoloading */
  739. regmap_write(backend->engine.regs, SUN4I_BACKEND_REGBUFFCTL_REG,
  740. SUN4I_BACKEND_REGBUFFCTL_AUTOLOAD_DIS);
  741. /* Enable the backend */
  742. regmap_write(backend->engine.regs, SUN4I_BACKEND_MODCTL_REG,
  743. SUN4I_BACKEND_MODCTL_DEBE_EN |
  744. SUN4I_BACKEND_MODCTL_START_CTL);
  745. /* Set output selection if needed */
  746. quirks = of_device_get_match_data(dev);
  747. if (quirks->needs_output_muxing) {
  748. /*
  749. * We assume there is no dynamic muxing of backends
  750. * and TCONs, so we select the backend with same ID.
  751. *
  752. * While dynamic selection might be interesting, since
  753. * the CRTC is tied to the TCON, while the layers are
  754. * tied to the backends, this means, we will need to
  755. * switch between groups of layers. There might not be
  756. * a way to represent this constraint in DRM.
  757. */
  758. regmap_update_bits(backend->engine.regs,
  759. SUN4I_BACKEND_MODCTL_REG,
  760. SUN4I_BACKEND_MODCTL_OUT_SEL,
  761. (backend->engine.id
  762. ? SUN4I_BACKEND_MODCTL_OUT_LCD1
  763. : SUN4I_BACKEND_MODCTL_OUT_LCD0));
  764. }
  765. backend->quirks = quirks;
  766. return 0;
  767. err_disable_ram_clk:
  768. clk_disable_unprepare(backend->ram_clk);
  769. err_disable_mod_clk:
  770. clk_rate_exclusive_put(backend->mod_clk);
  771. clk_disable_unprepare(backend->mod_clk);
  772. err_disable_bus_clk:
  773. clk_disable_unprepare(backend->bus_clk);
  774. err_assert_reset:
  775. reset_control_assert(backend->reset);
  776. return ret;
  777. }
  778. static void sun4i_backend_unbind(struct device *dev, struct device *master,
  779. void *data)
  780. {
  781. struct sun4i_backend *backend = dev_get_drvdata(dev);
  782. list_del(&backend->engine.list);
  783. if (of_device_is_compatible(dev->of_node,
  784. "allwinner,sun8i-a33-display-backend"))
  785. sun4i_backend_free_sat(dev);
  786. clk_disable_unprepare(backend->ram_clk);
  787. clk_rate_exclusive_put(backend->mod_clk);
  788. clk_disable_unprepare(backend->mod_clk);
  789. clk_disable_unprepare(backend->bus_clk);
  790. reset_control_assert(backend->reset);
  791. }
  792. static const struct component_ops sun4i_backend_ops = {
  793. .bind = sun4i_backend_bind,
  794. .unbind = sun4i_backend_unbind,
  795. };
  796. static int sun4i_backend_probe(struct platform_device *pdev)
  797. {
  798. return component_add(&pdev->dev, &sun4i_backend_ops);
  799. }
  800. static int sun4i_backend_remove(struct platform_device *pdev)
  801. {
  802. component_del(&pdev->dev, &sun4i_backend_ops);
  803. return 0;
  804. }
  805. static const struct sun4i_backend_quirks sun4i_backend_quirks = {
  806. .needs_output_muxing = true,
  807. };
  808. static const struct sun4i_backend_quirks sun5i_backend_quirks = {
  809. };
  810. static const struct sun4i_backend_quirks sun6i_backend_quirks = {
  811. };
  812. static const struct sun4i_backend_quirks sun7i_backend_quirks = {
  813. .needs_output_muxing = true,
  814. };
  815. static const struct sun4i_backend_quirks sun8i_a33_backend_quirks = {
  816. .supports_lowest_plane_alpha = true,
  817. };
  818. static const struct sun4i_backend_quirks sun9i_backend_quirks = {
  819. };
  820. static const struct of_device_id sun4i_backend_of_table[] = {
  821. {
  822. .compatible = "allwinner,sun4i-a10-display-backend",
  823. .data = &sun4i_backend_quirks,
  824. },
  825. {
  826. .compatible = "allwinner,sun5i-a13-display-backend",
  827. .data = &sun5i_backend_quirks,
  828. },
  829. {
  830. .compatible = "allwinner,sun6i-a31-display-backend",
  831. .data = &sun6i_backend_quirks,
  832. },
  833. {
  834. .compatible = "allwinner,sun7i-a20-display-backend",
  835. .data = &sun7i_backend_quirks,
  836. },
  837. {
  838. .compatible = "allwinner,sun8i-a23-display-backend",
  839. .data = &sun8i_a33_backend_quirks,
  840. },
  841. {
  842. .compatible = "allwinner,sun8i-a33-display-backend",
  843. .data = &sun8i_a33_backend_quirks,
  844. },
  845. {
  846. .compatible = "allwinner,sun9i-a80-display-backend",
  847. .data = &sun9i_backend_quirks,
  848. },
  849. { }
  850. };
  851. MODULE_DEVICE_TABLE(of, sun4i_backend_of_table);
  852. static struct platform_driver sun4i_backend_platform_driver = {
  853. .probe = sun4i_backend_probe,
  854. .remove = sun4i_backend_remove,
  855. .driver = {
  856. .name = "sun4i-backend",
  857. .of_match_table = sun4i_backend_of_table,
  858. },
  859. };
  860. module_platform_driver(sun4i_backend_platform_driver);
  861. MODULE_AUTHOR("Maxime Ripard <[email protected]>");
  862. MODULE_DESCRIPTION("Allwinner A10 Display Backend Driver");
  863. MODULE_LICENSE("GPL");