dc-plane.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright 2024 NXP
  4. */
  5. #include <linux/container_of.h>
  6. #include <drm/drm_atomic.h>
  7. #include <drm/drm_atomic_helper.h>
  8. #include <drm/drm_atomic_state_helper.h>
  9. #include <drm/drm_crtc.h>
  10. #include <drm/drm_drv.h>
  11. #include <drm/drm_fb_dma_helper.h>
  12. #include <drm/drm_fourcc.h>
  13. #include <drm/drm_framebuffer.h>
  14. #include <drm/drm_gem_atomic_helper.h>
  15. #include <drm/drm_plane_helper.h>
  16. #include <drm/drm_print.h>
  17. #include "dc-drv.h"
  18. #include "dc-fu.h"
  19. #include "dc-kms.h"
  20. #define DC_PLANE_MAX_PITCH 0x10000
  21. #define DC_PLANE_MAX_PIX_CNT 8192
  22. #define dc_plane_dbg(plane, fmt, ...) \
  23. do { \
  24. struct drm_plane *_plane = (plane); \
  25. drm_dbg_kms(_plane->dev, "[PLANE:%d:%s] " fmt, \
  26. _plane->base.id, _plane->name, ##__VA_ARGS__); \
  27. } while (0)
  28. static const uint32_t dc_plane_formats[] = {
  29. DRM_FORMAT_XRGB8888,
  30. };
  31. static const struct drm_plane_funcs dc_plane_funcs = {
  32. .update_plane = drm_atomic_helper_update_plane,
  33. .disable_plane = drm_atomic_helper_disable_plane,
  34. .destroy = drm_plane_cleanup,
  35. .reset = drm_atomic_helper_plane_reset,
  36. .atomic_duplicate_state = drm_atomic_helper_plane_duplicate_state,
  37. .atomic_destroy_state = drm_atomic_helper_plane_destroy_state,
  38. };
  39. static inline struct dc_plane *to_dc_plane(struct drm_plane *plane)
  40. {
  41. return container_of(plane, struct dc_plane, base);
  42. }
  43. static int dc_plane_check_max_source_resolution(struct drm_plane_state *state)
  44. {
  45. int src_h = drm_rect_height(&state->src) >> 16;
  46. int src_w = drm_rect_width(&state->src) >> 16;
  47. if (src_w > DC_PLANE_MAX_PIX_CNT || src_h > DC_PLANE_MAX_PIX_CNT) {
  48. dc_plane_dbg(state->plane, "invalid source resolution\n");
  49. return -EINVAL;
  50. }
  51. return 0;
  52. }
  53. static int dc_plane_check_fb(struct drm_plane_state *state)
  54. {
  55. struct drm_framebuffer *fb = state->fb;
  56. dma_addr_t baseaddr = drm_fb_dma_get_gem_addr(fb, state, 0);
  57. /* base address alignment */
  58. if (baseaddr & 0x3) {
  59. dc_plane_dbg(state->plane, "fb bad baddr alignment\n");
  60. return -EINVAL;
  61. }
  62. /* pitches[0] range */
  63. if (fb->pitches[0] > DC_PLANE_MAX_PITCH) {
  64. dc_plane_dbg(state->plane, "fb pitches[0] is out of range\n");
  65. return -EINVAL;
  66. }
  67. /* pitches[0] alignment */
  68. if (fb->pitches[0] & 0x3) {
  69. dc_plane_dbg(state->plane, "fb bad pitches[0] alignment\n");
  70. return -EINVAL;
  71. }
  72. return 0;
  73. }
  74. static int
  75. dc_plane_atomic_check(struct drm_plane *plane, struct drm_atomic_state *state)
  76. {
  77. struct drm_plane_state *plane_state =
  78. drm_atomic_get_new_plane_state(state, plane);
  79. struct drm_crtc_state *crtc_state;
  80. int ret;
  81. /* ok to disable */
  82. if (!plane_state->fb)
  83. return 0;
  84. if (!plane_state->crtc) {
  85. dc_plane_dbg(plane, "no CRTC in plane state\n");
  86. return -EINVAL;
  87. }
  88. crtc_state =
  89. drm_atomic_get_new_crtc_state(state, plane_state->crtc);
  90. if (WARN_ON(!crtc_state))
  91. return -EINVAL;
  92. ret = drm_atomic_helper_check_plane_state(plane_state, crtc_state,
  93. DRM_PLANE_NO_SCALING,
  94. DRM_PLANE_NO_SCALING,
  95. true, false);
  96. if (ret) {
  97. dc_plane_dbg(plane, "failed to check plane state: %d\n", ret);
  98. return ret;
  99. }
  100. ret = dc_plane_check_max_source_resolution(plane_state);
  101. if (ret)
  102. return ret;
  103. return dc_plane_check_fb(plane_state);
  104. }
  105. static void
  106. dc_plane_atomic_update(struct drm_plane *plane, struct drm_atomic_state *state)
  107. {
  108. struct drm_plane_state *new_state =
  109. drm_atomic_get_new_plane_state(state, plane);
  110. struct dc_plane *dplane = to_dc_plane(plane);
  111. struct drm_framebuffer *fb = new_state->fb;
  112. const struct dc_fu_ops *fu_ops;
  113. struct dc_lb *lb = dplane->lb;
  114. struct dc_fu *fu = dplane->fu;
  115. dma_addr_t baseaddr;
  116. int src_w, src_h;
  117. int idx;
  118. if (!drm_dev_enter(plane->dev, &idx))
  119. return;
  120. src_w = drm_rect_width(&new_state->src) >> 16;
  121. src_h = drm_rect_height(&new_state->src) >> 16;
  122. baseaddr = drm_fb_dma_get_gem_addr(fb, new_state, 0);
  123. fu_ops = dc_fu_get_ops(dplane->fu);
  124. fu_ops->set_layerblend(fu, lb);
  125. fu_ops->set_burstlength(fu, baseaddr);
  126. fu_ops->set_src_stride(fu, DC_FETCHUNIT_FRAC0, fb->pitches[0]);
  127. fu_ops->set_src_buf_dimensions(fu, DC_FETCHUNIT_FRAC0, src_w, src_h);
  128. fu_ops->set_fmt(fu, DC_FETCHUNIT_FRAC0, fb->format);
  129. fu_ops->set_framedimensions(fu, src_w, src_h);
  130. fu_ops->set_baseaddress(fu, DC_FETCHUNIT_FRAC0, baseaddr);
  131. fu_ops->enable_src_buf(fu, DC_FETCHUNIT_FRAC0);
  132. dc_plane_dbg(plane, "uses %s\n", fu_ops->get_name(fu));
  133. dc_lb_pec_dynamic_prim_sel(lb, dc_cf_get_link_id(dplane->cf));
  134. dc_lb_pec_dynamic_sec_sel(lb, fu_ops->get_link_id(fu));
  135. dc_lb_mode(lb, LB_BLEND);
  136. dc_lb_position(lb, new_state->dst.x1, new_state->dst.y1);
  137. dc_lb_pec_clken(lb, CLKEN_AUTOMATIC);
  138. dc_plane_dbg(plane, "uses LayerBlend%d\n", dc_lb_get_id(lb));
  139. /* set ExtDst's source to LayerBlend */
  140. dc_ed_pec_src_sel(dplane->ed, dc_lb_get_link_id(lb));
  141. drm_dev_exit(idx);
  142. }
  143. static void dc_plane_atomic_disable(struct drm_plane *plane,
  144. struct drm_atomic_state *state)
  145. {
  146. struct dc_plane *dplane = to_dc_plane(plane);
  147. const struct dc_fu_ops *fu_ops;
  148. int idx;
  149. if (!drm_dev_enter(plane->dev, &idx))
  150. return;
  151. /* disable fetchunit in shadow */
  152. fu_ops = dc_fu_get_ops(dplane->fu);
  153. fu_ops->disable_src_buf(dplane->fu, DC_FETCHUNIT_FRAC0);
  154. /* set ExtDst's source to ConstFrame */
  155. dc_ed_pec_src_sel(dplane->ed, dc_cf_get_link_id(dplane->cf));
  156. drm_dev_exit(idx);
  157. }
  158. static const struct drm_plane_helper_funcs dc_plane_helper_funcs = {
  159. .atomic_check = dc_plane_atomic_check,
  160. .atomic_update = dc_plane_atomic_update,
  161. .atomic_disable = dc_plane_atomic_disable,
  162. };
  163. int dc_plane_init(struct dc_drm_device *dc_drm, struct dc_plane *dc_plane)
  164. {
  165. struct drm_plane *plane = &dc_plane->base;
  166. int ret;
  167. ret = drm_universal_plane_init(&dc_drm->base, plane, 0, &dc_plane_funcs,
  168. dc_plane_formats,
  169. ARRAY_SIZE(dc_plane_formats),
  170. NULL, DRM_PLANE_TYPE_PRIMARY, NULL);
  171. if (ret)
  172. return ret;
  173. drm_plane_helper_add(plane, &dc_plane_helper_funcs);
  174. dc_plane->fu = dc_drm->pe->fu_disp[plane->index];
  175. dc_plane->cf = dc_drm->pe->cf_cont[plane->index];
  176. dc_plane->lb = dc_drm->pe->lb[plane->index];
  177. dc_plane->ed = dc_drm->pe->ed_cont[plane->index];
  178. return 0;
  179. }