dcn10_hubp.c 36 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174
  1. /*
  2. * Copyright 2012-15 Advanced Micro Devices, Inc.
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a
  5. * copy of this software and associated documentation files (the "Software"),
  6. * to deal in the Software without restriction, including without limitation
  7. * the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. * and/or sell copies of the Software, and to permit persons to whom the
  9. * Software is furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
  17. * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
  18. * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
  19. * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
  20. * OTHER DEALINGS IN THE SOFTWARE.
  21. *
  22. * Authors: AMD
  23. *
  24. */
  25. #include "dm_services.h"
  26. #include "dce_calcs.h"
  27. #include "reg_helper.h"
  28. #include "basics/conversion.h"
  29. #include "dcn10_hubp.h"
  30. #define REG(reg)\
  31. hubp1->hubp_regs->reg
  32. #define CTX \
  33. hubp1->base.ctx
  34. #undef FN
  35. #define FN(reg_name, field_name) \
  36. hubp1->hubp_shift->field_name, hubp1->hubp_mask->field_name
  37. void hubp1_set_blank(struct hubp *hubp, bool blank)
  38. {
  39. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  40. uint32_t blank_en = blank ? 1 : 0;
  41. REG_UPDATE_2(DCHUBP_CNTL,
  42. HUBP_BLANK_EN, blank_en,
  43. HUBP_TTU_DISABLE, blank_en);
  44. if (blank) {
  45. uint32_t reg_val = REG_READ(DCHUBP_CNTL);
  46. if (reg_val) {
  47. /* init sequence workaround: in case HUBP is
  48. * power gated, this wait would timeout.
  49. *
  50. * we just wrote reg_val to non-0, if it stay 0
  51. * it means HUBP is gated
  52. */
  53. REG_WAIT(DCHUBP_CNTL,
  54. HUBP_NO_OUTSTANDING_REQ, 1,
  55. 1, 200);
  56. }
  57. hubp->mpcc_id = 0xf;
  58. hubp->opp_id = 0xf;
  59. }
  60. }
  61. static void hubp1_disconnect(struct hubp *hubp)
  62. {
  63. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  64. REG_UPDATE(DCHUBP_CNTL,
  65. HUBP_TTU_DISABLE, 1);
  66. REG_UPDATE(CURSOR_CONTROL,
  67. CURSOR_ENABLE, 0);
  68. }
  69. static void hubp1_disable_control(struct hubp *hubp, bool disable_hubp)
  70. {
  71. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  72. uint32_t disable = disable_hubp ? 1 : 0;
  73. REG_UPDATE(DCHUBP_CNTL,
  74. HUBP_DISABLE, disable);
  75. }
  76. static unsigned int hubp1_get_underflow_status(struct hubp *hubp)
  77. {
  78. uint32_t hubp_underflow = 0;
  79. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  80. REG_GET(DCHUBP_CNTL,
  81. HUBP_UNDERFLOW_STATUS,
  82. &hubp_underflow);
  83. return hubp_underflow;
  84. }
  85. static void hubp1_set_hubp_blank_en(struct hubp *hubp, bool blank)
  86. {
  87. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  88. uint32_t blank_en = blank ? 1 : 0;
  89. REG_UPDATE(DCHUBP_CNTL, HUBP_BLANK_EN, blank_en);
  90. }
  91. static void hubp1_vready_workaround(struct hubp *hubp,
  92. struct _vcs_dpi_display_pipe_dest_params_st *pipe_dest)
  93. {
  94. uint32_t value = 0;
  95. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  96. /* set HBUBREQ_DEBUG_DB[12] = 1 */
  97. value = REG_READ(HUBPREQ_DEBUG_DB);
  98. /* hack mode disable */
  99. value |= 0x100;
  100. value &= ~0x1000;
  101. if ((pipe_dest->vstartup_start - 2*(pipe_dest->vready_offset+pipe_dest->vupdate_width
  102. + pipe_dest->vupdate_offset) / pipe_dest->htotal) <= pipe_dest->vblank_end) {
  103. /* if (eco_fix_needed(otg_global_sync_timing)
  104. * set HBUBREQ_DEBUG_DB[12] = 1 */
  105. value |= 0x1000;
  106. }
  107. REG_WRITE(HUBPREQ_DEBUG_DB, value);
  108. }
  109. void hubp1_program_tiling(
  110. struct hubp *hubp,
  111. const union dc_tiling_info *info,
  112. const enum surface_pixel_format pixel_format)
  113. {
  114. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  115. REG_UPDATE_6(DCSURF_ADDR_CONFIG,
  116. NUM_PIPES, log_2(info->gfx9.num_pipes),
  117. NUM_BANKS, log_2(info->gfx9.num_banks),
  118. PIPE_INTERLEAVE, info->gfx9.pipe_interleave,
  119. NUM_SE, log_2(info->gfx9.num_shader_engines),
  120. NUM_RB_PER_SE, log_2(info->gfx9.num_rb_per_se),
  121. MAX_COMPRESSED_FRAGS, log_2(info->gfx9.max_compressed_frags));
  122. REG_UPDATE_4(DCSURF_TILING_CONFIG,
  123. SW_MODE, info->gfx9.swizzle,
  124. META_LINEAR, info->gfx9.meta_linear,
  125. RB_ALIGNED, info->gfx9.rb_aligned,
  126. PIPE_ALIGNED, info->gfx9.pipe_aligned);
  127. }
  128. void hubp1_program_size_and_rotation(
  129. struct hubp *hubp,
  130. enum dc_rotation_angle rotation,
  131. enum surface_pixel_format format,
  132. const union plane_size *plane_size,
  133. struct dc_plane_dcc_param *dcc,
  134. bool horizontal_mirror)
  135. {
  136. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  137. uint32_t pitch, meta_pitch, pitch_c, meta_pitch_c, mirror;
  138. /* Program data and meta surface pitch (calculation from addrlib)
  139. * 444 or 420 luma
  140. */
  141. if (format >= SURFACE_PIXEL_FORMAT_VIDEO_BEGIN) {
  142. ASSERT(plane_size->video.chroma_pitch != 0);
  143. /* Chroma pitch zero can cause system hang! */
  144. pitch = plane_size->video.luma_pitch - 1;
  145. meta_pitch = dcc->video.meta_pitch_l - 1;
  146. pitch_c = plane_size->video.chroma_pitch - 1;
  147. meta_pitch_c = dcc->video.meta_pitch_c - 1;
  148. } else {
  149. pitch = plane_size->grph.surface_pitch - 1;
  150. meta_pitch = dcc->grph.meta_pitch - 1;
  151. pitch_c = 0;
  152. meta_pitch_c = 0;
  153. }
  154. if (!dcc->enable) {
  155. meta_pitch = 0;
  156. meta_pitch_c = 0;
  157. }
  158. REG_UPDATE_2(DCSURF_SURFACE_PITCH,
  159. PITCH, pitch, META_PITCH, meta_pitch);
  160. if (format >= SURFACE_PIXEL_FORMAT_VIDEO_BEGIN)
  161. REG_UPDATE_2(DCSURF_SURFACE_PITCH_C,
  162. PITCH_C, pitch_c, META_PITCH_C, meta_pitch_c);
  163. if (horizontal_mirror)
  164. mirror = 1;
  165. else
  166. mirror = 0;
  167. /* Program rotation angle and horz mirror - no mirror */
  168. if (rotation == ROTATION_ANGLE_0)
  169. REG_UPDATE_2(DCSURF_SURFACE_CONFIG,
  170. ROTATION_ANGLE, 0,
  171. H_MIRROR_EN, mirror);
  172. else if (rotation == ROTATION_ANGLE_90)
  173. REG_UPDATE_2(DCSURF_SURFACE_CONFIG,
  174. ROTATION_ANGLE, 1,
  175. H_MIRROR_EN, mirror);
  176. else if (rotation == ROTATION_ANGLE_180)
  177. REG_UPDATE_2(DCSURF_SURFACE_CONFIG,
  178. ROTATION_ANGLE, 2,
  179. H_MIRROR_EN, mirror);
  180. else if (rotation == ROTATION_ANGLE_270)
  181. REG_UPDATE_2(DCSURF_SURFACE_CONFIG,
  182. ROTATION_ANGLE, 3,
  183. H_MIRROR_EN, mirror);
  184. }
  185. void hubp1_program_pixel_format(
  186. struct hubp *hubp,
  187. enum surface_pixel_format format)
  188. {
  189. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  190. uint32_t red_bar = 3;
  191. uint32_t blue_bar = 2;
  192. /* swap for ABGR format */
  193. if (format == SURFACE_PIXEL_FORMAT_GRPH_ABGR8888
  194. || format == SURFACE_PIXEL_FORMAT_GRPH_ABGR2101010
  195. || format == SURFACE_PIXEL_FORMAT_GRPH_ABGR2101010_XR_BIAS
  196. || format == SURFACE_PIXEL_FORMAT_GRPH_ABGR16161616F) {
  197. red_bar = 2;
  198. blue_bar = 3;
  199. }
  200. REG_UPDATE_2(HUBPRET_CONTROL,
  201. CROSSBAR_SRC_CB_B, blue_bar,
  202. CROSSBAR_SRC_CR_R, red_bar);
  203. /* Mapping is same as ipp programming (cnvc) */
  204. switch (format) {
  205. case SURFACE_PIXEL_FORMAT_GRPH_ARGB1555:
  206. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  207. SURFACE_PIXEL_FORMAT, 1);
  208. break;
  209. case SURFACE_PIXEL_FORMAT_GRPH_RGB565:
  210. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  211. SURFACE_PIXEL_FORMAT, 3);
  212. break;
  213. case SURFACE_PIXEL_FORMAT_GRPH_ARGB8888:
  214. case SURFACE_PIXEL_FORMAT_GRPH_ABGR8888:
  215. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  216. SURFACE_PIXEL_FORMAT, 8);
  217. break;
  218. case SURFACE_PIXEL_FORMAT_GRPH_ARGB2101010:
  219. case SURFACE_PIXEL_FORMAT_GRPH_ABGR2101010:
  220. case SURFACE_PIXEL_FORMAT_GRPH_ABGR2101010_XR_BIAS:
  221. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  222. SURFACE_PIXEL_FORMAT, 10);
  223. break;
  224. case SURFACE_PIXEL_FORMAT_GRPH_ARGB16161616:
  225. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  226. SURFACE_PIXEL_FORMAT, 22);
  227. break;
  228. case SURFACE_PIXEL_FORMAT_GRPH_ARGB16161616F:
  229. case SURFACE_PIXEL_FORMAT_GRPH_ABGR16161616F:/*we use crossbar already*/
  230. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  231. SURFACE_PIXEL_FORMAT, 24);
  232. break;
  233. case SURFACE_PIXEL_FORMAT_VIDEO_420_YCbCr:
  234. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  235. SURFACE_PIXEL_FORMAT, 65);
  236. break;
  237. case SURFACE_PIXEL_FORMAT_VIDEO_420_YCrCb:
  238. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  239. SURFACE_PIXEL_FORMAT, 64);
  240. break;
  241. case SURFACE_PIXEL_FORMAT_VIDEO_420_10bpc_YCbCr:
  242. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  243. SURFACE_PIXEL_FORMAT, 67);
  244. break;
  245. case SURFACE_PIXEL_FORMAT_VIDEO_420_10bpc_YCrCb:
  246. REG_UPDATE(DCSURF_SURFACE_CONFIG,
  247. SURFACE_PIXEL_FORMAT, 66);
  248. break;
  249. default:
  250. BREAK_TO_DEBUGGER();
  251. break;
  252. }
  253. /* don't see the need of program the xbar in DCN 1.0 */
  254. }
  255. bool hubp1_program_surface_flip_and_addr(
  256. struct hubp *hubp,
  257. const struct dc_plane_address *address,
  258. bool flip_immediate)
  259. {
  260. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  261. /* program flip type */
  262. REG_SET(DCSURF_FLIP_CONTROL, 0,
  263. SURFACE_FLIP_TYPE, flip_immediate);
  264. /* HW automatically latch rest of address register on write to
  265. * DCSURF_PRIMARY_SURFACE_ADDRESS if SURFACE_UPDATE_LOCK is not used
  266. *
  267. * program high first and then the low addr, order matters!
  268. */
  269. switch (address->type) {
  270. case PLN_ADDR_TYPE_GRAPHICS:
  271. /* DCN1.0 does not support const color
  272. * TODO: program DCHUBBUB_RET_PATH_DCC_CFGx_0/1
  273. * base on address->grph.dcc_const_color
  274. * x = 0, 2, 4, 6 for pipe 0, 1, 2, 3 for rgb and luma
  275. * x = 1, 3, 5, 7 for pipe 0, 1, 2, 3 for chroma
  276. */
  277. if (address->grph.addr.quad_part == 0)
  278. break;
  279. REG_UPDATE_2(DCSURF_SURFACE_CONTROL,
  280. PRIMARY_SURFACE_TMZ, address->tmz_surface,
  281. PRIMARY_META_SURFACE_TMZ, address->tmz_surface);
  282. if (address->grph.meta_addr.quad_part != 0) {
  283. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS_HIGH, 0,
  284. PRIMARY_META_SURFACE_ADDRESS_HIGH,
  285. address->grph.meta_addr.high_part);
  286. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS, 0,
  287. PRIMARY_META_SURFACE_ADDRESS,
  288. address->grph.meta_addr.low_part);
  289. }
  290. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS_HIGH, 0,
  291. PRIMARY_SURFACE_ADDRESS_HIGH,
  292. address->grph.addr.high_part);
  293. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS, 0,
  294. PRIMARY_SURFACE_ADDRESS,
  295. address->grph.addr.low_part);
  296. break;
  297. case PLN_ADDR_TYPE_VIDEO_PROGRESSIVE:
  298. if (address->video_progressive.luma_addr.quad_part == 0
  299. || address->video_progressive.chroma_addr.quad_part == 0)
  300. break;
  301. REG_UPDATE_4(DCSURF_SURFACE_CONTROL,
  302. PRIMARY_SURFACE_TMZ, address->tmz_surface,
  303. PRIMARY_SURFACE_TMZ_C, address->tmz_surface,
  304. PRIMARY_META_SURFACE_TMZ, address->tmz_surface,
  305. PRIMARY_META_SURFACE_TMZ_C, address->tmz_surface);
  306. if (address->video_progressive.luma_meta_addr.quad_part != 0) {
  307. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS_HIGH_C, 0,
  308. PRIMARY_META_SURFACE_ADDRESS_HIGH_C,
  309. address->video_progressive.chroma_meta_addr.high_part);
  310. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS_C, 0,
  311. PRIMARY_META_SURFACE_ADDRESS_C,
  312. address->video_progressive.chroma_meta_addr.low_part);
  313. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS_HIGH, 0,
  314. PRIMARY_META_SURFACE_ADDRESS_HIGH,
  315. address->video_progressive.luma_meta_addr.high_part);
  316. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS, 0,
  317. PRIMARY_META_SURFACE_ADDRESS,
  318. address->video_progressive.luma_meta_addr.low_part);
  319. }
  320. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS_HIGH_C, 0,
  321. PRIMARY_SURFACE_ADDRESS_HIGH_C,
  322. address->video_progressive.chroma_addr.high_part);
  323. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS_C, 0,
  324. PRIMARY_SURFACE_ADDRESS_C,
  325. address->video_progressive.chroma_addr.low_part);
  326. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS_HIGH, 0,
  327. PRIMARY_SURFACE_ADDRESS_HIGH,
  328. address->video_progressive.luma_addr.high_part);
  329. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS, 0,
  330. PRIMARY_SURFACE_ADDRESS,
  331. address->video_progressive.luma_addr.low_part);
  332. break;
  333. case PLN_ADDR_TYPE_GRPH_STEREO:
  334. if (address->grph_stereo.left_addr.quad_part == 0)
  335. break;
  336. if (address->grph_stereo.right_addr.quad_part == 0)
  337. break;
  338. REG_UPDATE_8(DCSURF_SURFACE_CONTROL,
  339. PRIMARY_SURFACE_TMZ, address->tmz_surface,
  340. PRIMARY_SURFACE_TMZ_C, address->tmz_surface,
  341. PRIMARY_META_SURFACE_TMZ, address->tmz_surface,
  342. PRIMARY_META_SURFACE_TMZ_C, address->tmz_surface,
  343. SECONDARY_SURFACE_TMZ, address->tmz_surface,
  344. SECONDARY_SURFACE_TMZ_C, address->tmz_surface,
  345. SECONDARY_META_SURFACE_TMZ, address->tmz_surface,
  346. SECONDARY_META_SURFACE_TMZ_C, address->tmz_surface);
  347. if (address->grph_stereo.right_meta_addr.quad_part != 0) {
  348. REG_SET(DCSURF_SECONDARY_META_SURFACE_ADDRESS_HIGH, 0,
  349. SECONDARY_META_SURFACE_ADDRESS_HIGH,
  350. address->grph_stereo.right_meta_addr.high_part);
  351. REG_SET(DCSURF_SECONDARY_META_SURFACE_ADDRESS, 0,
  352. SECONDARY_META_SURFACE_ADDRESS,
  353. address->grph_stereo.right_meta_addr.low_part);
  354. }
  355. if (address->grph_stereo.left_meta_addr.quad_part != 0) {
  356. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS_HIGH, 0,
  357. PRIMARY_META_SURFACE_ADDRESS_HIGH,
  358. address->grph_stereo.left_meta_addr.high_part);
  359. REG_SET(DCSURF_PRIMARY_META_SURFACE_ADDRESS, 0,
  360. PRIMARY_META_SURFACE_ADDRESS,
  361. address->grph_stereo.left_meta_addr.low_part);
  362. }
  363. REG_SET(DCSURF_SECONDARY_SURFACE_ADDRESS_HIGH, 0,
  364. SECONDARY_SURFACE_ADDRESS_HIGH,
  365. address->grph_stereo.right_addr.high_part);
  366. REG_SET(DCSURF_SECONDARY_SURFACE_ADDRESS, 0,
  367. SECONDARY_SURFACE_ADDRESS,
  368. address->grph_stereo.right_addr.low_part);
  369. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS_HIGH, 0,
  370. PRIMARY_SURFACE_ADDRESS_HIGH,
  371. address->grph_stereo.left_addr.high_part);
  372. REG_SET(DCSURF_PRIMARY_SURFACE_ADDRESS, 0,
  373. PRIMARY_SURFACE_ADDRESS,
  374. address->grph_stereo.left_addr.low_part);
  375. break;
  376. default:
  377. BREAK_TO_DEBUGGER();
  378. break;
  379. }
  380. hubp->request_address = *address;
  381. if (flip_immediate)
  382. hubp->current_address = *address;
  383. return true;
  384. }
  385. void hubp1_dcc_control(struct hubp *hubp, bool enable,
  386. bool independent_64b_blks)
  387. {
  388. uint32_t dcc_en = enable ? 1 : 0;
  389. uint32_t dcc_ind_64b_blk = independent_64b_blks ? 1 : 0;
  390. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  391. REG_UPDATE_4(DCSURF_SURFACE_CONTROL,
  392. PRIMARY_SURFACE_DCC_EN, dcc_en,
  393. PRIMARY_SURFACE_DCC_IND_64B_BLK, dcc_ind_64b_blk,
  394. SECONDARY_SURFACE_DCC_EN, dcc_en,
  395. SECONDARY_SURFACE_DCC_IND_64B_BLK, dcc_ind_64b_blk);
  396. }
  397. void hubp1_program_surface_config(
  398. struct hubp *hubp,
  399. enum surface_pixel_format format,
  400. union dc_tiling_info *tiling_info,
  401. union plane_size *plane_size,
  402. enum dc_rotation_angle rotation,
  403. struct dc_plane_dcc_param *dcc,
  404. bool horizontal_mirror)
  405. {
  406. hubp1_dcc_control(hubp, dcc->enable, dcc->grph.independent_64b_blks);
  407. hubp1_program_tiling(hubp, tiling_info, format);
  408. hubp1_program_size_and_rotation(
  409. hubp, rotation, format, plane_size, dcc, horizontal_mirror);
  410. hubp1_program_pixel_format(hubp, format);
  411. }
  412. void hubp1_program_requestor(
  413. struct hubp *hubp,
  414. struct _vcs_dpi_display_rq_regs_st *rq_regs)
  415. {
  416. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  417. REG_UPDATE(HUBPRET_CONTROL,
  418. DET_BUF_PLANE1_BASE_ADDRESS, rq_regs->plane1_base_address);
  419. REG_SET_4(DCN_EXPANSION_MODE, 0,
  420. DRQ_EXPANSION_MODE, rq_regs->drq_expansion_mode,
  421. PRQ_EXPANSION_MODE, rq_regs->prq_expansion_mode,
  422. MRQ_EXPANSION_MODE, rq_regs->mrq_expansion_mode,
  423. CRQ_EXPANSION_MODE, rq_regs->crq_expansion_mode);
  424. REG_SET_8(DCHUBP_REQ_SIZE_CONFIG, 0,
  425. CHUNK_SIZE, rq_regs->rq_regs_l.chunk_size,
  426. MIN_CHUNK_SIZE, rq_regs->rq_regs_l.min_chunk_size,
  427. META_CHUNK_SIZE, rq_regs->rq_regs_l.meta_chunk_size,
  428. MIN_META_CHUNK_SIZE, rq_regs->rq_regs_l.min_meta_chunk_size,
  429. DPTE_GROUP_SIZE, rq_regs->rq_regs_l.dpte_group_size,
  430. MPTE_GROUP_SIZE, rq_regs->rq_regs_l.mpte_group_size,
  431. SWATH_HEIGHT, rq_regs->rq_regs_l.swath_height,
  432. PTE_ROW_HEIGHT_LINEAR, rq_regs->rq_regs_l.pte_row_height_linear);
  433. REG_SET_8(DCHUBP_REQ_SIZE_CONFIG_C, 0,
  434. CHUNK_SIZE_C, rq_regs->rq_regs_c.chunk_size,
  435. MIN_CHUNK_SIZE_C, rq_regs->rq_regs_c.min_chunk_size,
  436. META_CHUNK_SIZE_C, rq_regs->rq_regs_c.meta_chunk_size,
  437. MIN_META_CHUNK_SIZE_C, rq_regs->rq_regs_c.min_meta_chunk_size,
  438. DPTE_GROUP_SIZE_C, rq_regs->rq_regs_c.dpte_group_size,
  439. MPTE_GROUP_SIZE_C, rq_regs->rq_regs_c.mpte_group_size,
  440. SWATH_HEIGHT_C, rq_regs->rq_regs_c.swath_height,
  441. PTE_ROW_HEIGHT_LINEAR_C, rq_regs->rq_regs_c.pte_row_height_linear);
  442. }
  443. void hubp1_program_deadline(
  444. struct hubp *hubp,
  445. struct _vcs_dpi_display_dlg_regs_st *dlg_attr,
  446. struct _vcs_dpi_display_ttu_regs_st *ttu_attr)
  447. {
  448. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  449. /* DLG - Per hubp */
  450. REG_SET_2(BLANK_OFFSET_0, 0,
  451. REFCYC_H_BLANK_END, dlg_attr->refcyc_h_blank_end,
  452. DLG_V_BLANK_END, dlg_attr->dlg_vblank_end);
  453. REG_SET(BLANK_OFFSET_1, 0,
  454. MIN_DST_Y_NEXT_START, dlg_attr->min_dst_y_next_start);
  455. REG_SET(DST_DIMENSIONS, 0,
  456. REFCYC_PER_HTOTAL, dlg_attr->refcyc_per_htotal);
  457. REG_SET_2(DST_AFTER_SCALER, 0,
  458. REFCYC_X_AFTER_SCALER, dlg_attr->refcyc_x_after_scaler,
  459. DST_Y_AFTER_SCALER, dlg_attr->dst_y_after_scaler);
  460. if (REG(PREFETCH_SETTINS))
  461. REG_SET_2(PREFETCH_SETTINS, 0,
  462. DST_Y_PREFETCH, dlg_attr->dst_y_prefetch,
  463. VRATIO_PREFETCH, dlg_attr->vratio_prefetch);
  464. else
  465. REG_SET_2(PREFETCH_SETTINGS, 0,
  466. DST_Y_PREFETCH, dlg_attr->dst_y_prefetch,
  467. VRATIO_PREFETCH, dlg_attr->vratio_prefetch);
  468. REG_SET_2(VBLANK_PARAMETERS_0, 0,
  469. DST_Y_PER_VM_VBLANK, dlg_attr->dst_y_per_vm_vblank,
  470. DST_Y_PER_ROW_VBLANK, dlg_attr->dst_y_per_row_vblank);
  471. REG_SET(REF_FREQ_TO_PIX_FREQ, 0,
  472. REF_FREQ_TO_PIX_FREQ, dlg_attr->ref_freq_to_pix_freq);
  473. /* DLG - Per luma/chroma */
  474. REG_SET(VBLANK_PARAMETERS_1, 0,
  475. REFCYC_PER_PTE_GROUP_VBLANK_L, dlg_attr->refcyc_per_pte_group_vblank_l);
  476. REG_SET(VBLANK_PARAMETERS_3, 0,
  477. REFCYC_PER_META_CHUNK_VBLANK_L, dlg_attr->refcyc_per_meta_chunk_vblank_l);
  478. if (REG(NOM_PARAMETERS_0))
  479. REG_SET(NOM_PARAMETERS_0, 0,
  480. DST_Y_PER_PTE_ROW_NOM_L, dlg_attr->dst_y_per_pte_row_nom_l);
  481. if (REG(NOM_PARAMETERS_1))
  482. REG_SET(NOM_PARAMETERS_1, 0,
  483. REFCYC_PER_PTE_GROUP_NOM_L, dlg_attr->refcyc_per_pte_group_nom_l);
  484. REG_SET(NOM_PARAMETERS_4, 0,
  485. DST_Y_PER_META_ROW_NOM_L, dlg_attr->dst_y_per_meta_row_nom_l);
  486. REG_SET(NOM_PARAMETERS_5, 0,
  487. REFCYC_PER_META_CHUNK_NOM_L, dlg_attr->refcyc_per_meta_chunk_nom_l);
  488. REG_SET_2(PER_LINE_DELIVERY_PRE, 0,
  489. REFCYC_PER_LINE_DELIVERY_PRE_L, dlg_attr->refcyc_per_line_delivery_pre_l,
  490. REFCYC_PER_LINE_DELIVERY_PRE_C, dlg_attr->refcyc_per_line_delivery_pre_c);
  491. REG_SET_2(PER_LINE_DELIVERY, 0,
  492. REFCYC_PER_LINE_DELIVERY_L, dlg_attr->refcyc_per_line_delivery_l,
  493. REFCYC_PER_LINE_DELIVERY_C, dlg_attr->refcyc_per_line_delivery_c);
  494. if (REG(PREFETCH_SETTINS_C))
  495. REG_SET(PREFETCH_SETTINS_C, 0,
  496. VRATIO_PREFETCH_C, dlg_attr->vratio_prefetch_c);
  497. else
  498. REG_SET(PREFETCH_SETTINGS_C, 0,
  499. VRATIO_PREFETCH_C, dlg_attr->vratio_prefetch_c);
  500. REG_SET(VBLANK_PARAMETERS_2, 0,
  501. REFCYC_PER_PTE_GROUP_VBLANK_C, dlg_attr->refcyc_per_pte_group_vblank_c);
  502. REG_SET(VBLANK_PARAMETERS_4, 0,
  503. REFCYC_PER_META_CHUNK_VBLANK_C, dlg_attr->refcyc_per_meta_chunk_vblank_c);
  504. if (REG(NOM_PARAMETERS_2))
  505. REG_SET(NOM_PARAMETERS_2, 0,
  506. DST_Y_PER_PTE_ROW_NOM_C, dlg_attr->dst_y_per_pte_row_nom_c);
  507. if (REG(NOM_PARAMETERS_3))
  508. REG_SET(NOM_PARAMETERS_3, 0,
  509. REFCYC_PER_PTE_GROUP_NOM_C, dlg_attr->refcyc_per_pte_group_nom_c);
  510. REG_SET(NOM_PARAMETERS_6, 0,
  511. DST_Y_PER_META_ROW_NOM_C, dlg_attr->dst_y_per_meta_row_nom_c);
  512. REG_SET(NOM_PARAMETERS_7, 0,
  513. REFCYC_PER_META_CHUNK_NOM_C, dlg_attr->refcyc_per_meta_chunk_nom_c);
  514. /* TTU - per hubp */
  515. REG_SET_2(DCN_TTU_QOS_WM, 0,
  516. QoS_LEVEL_LOW_WM, ttu_attr->qos_level_low_wm,
  517. QoS_LEVEL_HIGH_WM, ttu_attr->qos_level_high_wm);
  518. REG_SET_2(DCN_GLOBAL_TTU_CNTL, 0,
  519. MIN_TTU_VBLANK, ttu_attr->min_ttu_vblank,
  520. QoS_LEVEL_FLIP, ttu_attr->qos_level_flip);
  521. /* TTU - per luma/chroma */
  522. /* Assumed surf0 is luma and 1 is chroma */
  523. REG_SET_3(DCN_SURF0_TTU_CNTL0, 0,
  524. REFCYC_PER_REQ_DELIVERY, ttu_attr->refcyc_per_req_delivery_l,
  525. QoS_LEVEL_FIXED, ttu_attr->qos_level_fixed_l,
  526. QoS_RAMP_DISABLE, ttu_attr->qos_ramp_disable_l);
  527. REG_SET(DCN_SURF0_TTU_CNTL1, 0,
  528. REFCYC_PER_REQ_DELIVERY_PRE,
  529. ttu_attr->refcyc_per_req_delivery_pre_l);
  530. REG_SET_3(DCN_SURF1_TTU_CNTL0, 0,
  531. REFCYC_PER_REQ_DELIVERY, ttu_attr->refcyc_per_req_delivery_c,
  532. QoS_LEVEL_FIXED, ttu_attr->qos_level_fixed_c,
  533. QoS_RAMP_DISABLE, ttu_attr->qos_ramp_disable_c);
  534. REG_SET(DCN_SURF1_TTU_CNTL1, 0,
  535. REFCYC_PER_REQ_DELIVERY_PRE,
  536. ttu_attr->refcyc_per_req_delivery_pre_c);
  537. REG_SET_3(DCN_CUR0_TTU_CNTL0, 0,
  538. REFCYC_PER_REQ_DELIVERY, ttu_attr->refcyc_per_req_delivery_cur0,
  539. QoS_LEVEL_FIXED, ttu_attr->qos_level_fixed_cur0,
  540. QoS_RAMP_DISABLE, ttu_attr->qos_ramp_disable_cur0);
  541. REG_SET(DCN_CUR0_TTU_CNTL1, 0,
  542. REFCYC_PER_REQ_DELIVERY_PRE, ttu_attr->refcyc_per_req_delivery_pre_cur0);
  543. }
  544. static void hubp1_setup(
  545. struct hubp *hubp,
  546. struct _vcs_dpi_display_dlg_regs_st *dlg_attr,
  547. struct _vcs_dpi_display_ttu_regs_st *ttu_attr,
  548. struct _vcs_dpi_display_rq_regs_st *rq_regs,
  549. struct _vcs_dpi_display_pipe_dest_params_st *pipe_dest)
  550. {
  551. /* otg is locked when this func is called. Register are double buffered.
  552. * disable the requestors is not needed
  553. */
  554. hubp1_program_requestor(hubp, rq_regs);
  555. hubp1_program_deadline(hubp, dlg_attr, ttu_attr);
  556. hubp1_vready_workaround(hubp, pipe_dest);
  557. }
  558. bool hubp1_is_flip_pending(struct hubp *hubp)
  559. {
  560. uint32_t flip_pending = 0;
  561. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  562. struct dc_plane_address earliest_inuse_address;
  563. REG_GET(DCSURF_FLIP_CONTROL,
  564. SURFACE_FLIP_PENDING, &flip_pending);
  565. REG_GET(DCSURF_SURFACE_EARLIEST_INUSE,
  566. SURFACE_EARLIEST_INUSE_ADDRESS, &earliest_inuse_address.grph.addr.low_part);
  567. REG_GET(DCSURF_SURFACE_EARLIEST_INUSE_HIGH,
  568. SURFACE_EARLIEST_INUSE_ADDRESS_HIGH, &earliest_inuse_address.grph.addr.high_part);
  569. if (flip_pending)
  570. return true;
  571. if (earliest_inuse_address.grph.addr.quad_part != hubp->request_address.grph.addr.quad_part)
  572. return true;
  573. hubp->current_address = hubp->request_address;
  574. return false;
  575. }
  576. uint32_t aperture_default_system = 1;
  577. uint32_t context0_default_system; /* = 0;*/
  578. static void hubp1_set_vm_system_aperture_settings(struct hubp *hubp,
  579. struct vm_system_aperture_param *apt)
  580. {
  581. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  582. PHYSICAL_ADDRESS_LOC mc_vm_apt_default;
  583. PHYSICAL_ADDRESS_LOC mc_vm_apt_low;
  584. PHYSICAL_ADDRESS_LOC mc_vm_apt_high;
  585. mc_vm_apt_default.quad_part = apt->sys_default.quad_part >> 12;
  586. mc_vm_apt_low.quad_part = apt->sys_low.quad_part >> 12;
  587. mc_vm_apt_high.quad_part = apt->sys_high.quad_part >> 12;
  588. REG_SET_2(DCN_VM_SYSTEM_APERTURE_DEFAULT_ADDR_MSB, 0,
  589. MC_VM_SYSTEM_APERTURE_DEFAULT_SYSTEM, aperture_default_system, /* 1 = system physical memory */
  590. MC_VM_SYSTEM_APERTURE_DEFAULT_ADDR_MSB, mc_vm_apt_default.high_part);
  591. REG_SET(DCN_VM_SYSTEM_APERTURE_DEFAULT_ADDR_LSB, 0,
  592. MC_VM_SYSTEM_APERTURE_DEFAULT_ADDR_LSB, mc_vm_apt_default.low_part);
  593. REG_SET(DCN_VM_SYSTEM_APERTURE_LOW_ADDR_MSB, 0,
  594. MC_VM_SYSTEM_APERTURE_LOW_ADDR_MSB, mc_vm_apt_low.high_part);
  595. REG_SET(DCN_VM_SYSTEM_APERTURE_LOW_ADDR_LSB, 0,
  596. MC_VM_SYSTEM_APERTURE_LOW_ADDR_LSB, mc_vm_apt_low.low_part);
  597. REG_SET(DCN_VM_SYSTEM_APERTURE_HIGH_ADDR_MSB, 0,
  598. MC_VM_SYSTEM_APERTURE_HIGH_ADDR_MSB, mc_vm_apt_high.high_part);
  599. REG_SET(DCN_VM_SYSTEM_APERTURE_HIGH_ADDR_LSB, 0,
  600. MC_VM_SYSTEM_APERTURE_HIGH_ADDR_LSB, mc_vm_apt_high.low_part);
  601. }
  602. static void hubp1_set_vm_context0_settings(struct hubp *hubp,
  603. const struct vm_context0_param *vm0)
  604. {
  605. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  606. /* pte base */
  607. REG_SET(DCN_VM_CONTEXT0_PAGE_TABLE_BASE_ADDR_MSB, 0,
  608. VM_CONTEXT0_PAGE_TABLE_BASE_ADDR_MSB, vm0->pte_base.high_part);
  609. REG_SET(DCN_VM_CONTEXT0_PAGE_TABLE_BASE_ADDR_LSB, 0,
  610. VM_CONTEXT0_PAGE_TABLE_BASE_ADDR_LSB, vm0->pte_base.low_part);
  611. /* pte start */
  612. REG_SET(DCN_VM_CONTEXT0_PAGE_TABLE_START_ADDR_MSB, 0,
  613. VM_CONTEXT0_PAGE_TABLE_START_ADDR_MSB, vm0->pte_start.high_part);
  614. REG_SET(DCN_VM_CONTEXT0_PAGE_TABLE_START_ADDR_LSB, 0,
  615. VM_CONTEXT0_PAGE_TABLE_START_ADDR_LSB, vm0->pte_start.low_part);
  616. /* pte end */
  617. REG_SET(DCN_VM_CONTEXT0_PAGE_TABLE_END_ADDR_MSB, 0,
  618. VM_CONTEXT0_PAGE_TABLE_END_ADDR_MSB, vm0->pte_end.high_part);
  619. REG_SET(DCN_VM_CONTEXT0_PAGE_TABLE_END_ADDR_LSB, 0,
  620. VM_CONTEXT0_PAGE_TABLE_END_ADDR_LSB, vm0->pte_end.low_part);
  621. /* fault handling */
  622. REG_SET_2(DCN_VM_CONTEXT0_PROTECTION_FAULT_DEFAULT_ADDR_MSB, 0,
  623. VM_CONTEXT0_PROTECTION_FAULT_DEFAULT_ADDR_MSB, vm0->fault_default.high_part,
  624. VM_CONTEXT0_PROTECTION_FAULT_DEFAULT_SYSTEM, context0_default_system);
  625. REG_SET(DCN_VM_CONTEXT0_PROTECTION_FAULT_DEFAULT_ADDR_LSB, 0,
  626. VM_CONTEXT0_PROTECTION_FAULT_DEFAULT_ADDR_LSB, vm0->fault_default.low_part);
  627. /* control: enable VM PTE*/
  628. REG_SET_2(DCN_VM_MX_L1_TLB_CNTL, 0,
  629. ENABLE_L1_TLB, 1,
  630. SYSTEM_ACCESS_MODE, 3);
  631. }
  632. void min_set_viewport(
  633. struct hubp *hubp,
  634. const struct rect *viewport,
  635. const struct rect *viewport_c)
  636. {
  637. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  638. REG_SET_2(DCSURF_PRI_VIEWPORT_DIMENSION, 0,
  639. PRI_VIEWPORT_WIDTH, viewport->width,
  640. PRI_VIEWPORT_HEIGHT, viewport->height);
  641. REG_SET_2(DCSURF_PRI_VIEWPORT_START, 0,
  642. PRI_VIEWPORT_X_START, viewport->x,
  643. PRI_VIEWPORT_Y_START, viewport->y);
  644. /*for stereo*/
  645. REG_SET_2(DCSURF_SEC_VIEWPORT_DIMENSION, 0,
  646. SEC_VIEWPORT_WIDTH, viewport->width,
  647. SEC_VIEWPORT_HEIGHT, viewport->height);
  648. REG_SET_2(DCSURF_SEC_VIEWPORT_START, 0,
  649. SEC_VIEWPORT_X_START, viewport->x,
  650. SEC_VIEWPORT_Y_START, viewport->y);
  651. /* DC supports NV12 only at the moment */
  652. REG_SET_2(DCSURF_PRI_VIEWPORT_DIMENSION_C, 0,
  653. PRI_VIEWPORT_WIDTH_C, viewport_c->width,
  654. PRI_VIEWPORT_HEIGHT_C, viewport_c->height);
  655. REG_SET_2(DCSURF_PRI_VIEWPORT_START_C, 0,
  656. PRI_VIEWPORT_X_START_C, viewport_c->x,
  657. PRI_VIEWPORT_Y_START_C, viewport_c->y);
  658. }
  659. void hubp1_read_state(struct hubp *hubp)
  660. {
  661. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  662. struct dcn_hubp_state *s = &hubp1->state;
  663. struct _vcs_dpi_display_dlg_regs_st *dlg_attr = &s->dlg_attr;
  664. struct _vcs_dpi_display_ttu_regs_st *ttu_attr = &s->ttu_attr;
  665. struct _vcs_dpi_display_rq_regs_st *rq_regs = &s->rq_regs;
  666. /* Requester */
  667. REG_GET(HUBPRET_CONTROL,
  668. DET_BUF_PLANE1_BASE_ADDRESS, &rq_regs->plane1_base_address);
  669. REG_GET_4(DCN_EXPANSION_MODE,
  670. DRQ_EXPANSION_MODE, &rq_regs->drq_expansion_mode,
  671. PRQ_EXPANSION_MODE, &rq_regs->prq_expansion_mode,
  672. MRQ_EXPANSION_MODE, &rq_regs->mrq_expansion_mode,
  673. CRQ_EXPANSION_MODE, &rq_regs->crq_expansion_mode);
  674. REG_GET_8(DCHUBP_REQ_SIZE_CONFIG,
  675. CHUNK_SIZE, &rq_regs->rq_regs_l.chunk_size,
  676. MIN_CHUNK_SIZE, &rq_regs->rq_regs_l.min_chunk_size,
  677. META_CHUNK_SIZE, &rq_regs->rq_regs_l.meta_chunk_size,
  678. MIN_META_CHUNK_SIZE, &rq_regs->rq_regs_l.min_meta_chunk_size,
  679. DPTE_GROUP_SIZE, &rq_regs->rq_regs_l.dpte_group_size,
  680. MPTE_GROUP_SIZE, &rq_regs->rq_regs_l.mpte_group_size,
  681. SWATH_HEIGHT, &rq_regs->rq_regs_l.swath_height,
  682. PTE_ROW_HEIGHT_LINEAR, &rq_regs->rq_regs_l.pte_row_height_linear);
  683. REG_GET_8(DCHUBP_REQ_SIZE_CONFIG_C,
  684. CHUNK_SIZE_C, &rq_regs->rq_regs_c.chunk_size,
  685. MIN_CHUNK_SIZE_C, &rq_regs->rq_regs_c.min_chunk_size,
  686. META_CHUNK_SIZE_C, &rq_regs->rq_regs_c.meta_chunk_size,
  687. MIN_META_CHUNK_SIZE_C, &rq_regs->rq_regs_c.min_meta_chunk_size,
  688. DPTE_GROUP_SIZE_C, &rq_regs->rq_regs_c.dpte_group_size,
  689. MPTE_GROUP_SIZE_C, &rq_regs->rq_regs_c.mpte_group_size,
  690. SWATH_HEIGHT_C, &rq_regs->rq_regs_c.swath_height,
  691. PTE_ROW_HEIGHT_LINEAR_C, &rq_regs->rq_regs_c.pte_row_height_linear);
  692. /* DLG - Per hubp */
  693. REG_GET_2(BLANK_OFFSET_0,
  694. REFCYC_H_BLANK_END, &dlg_attr->refcyc_h_blank_end,
  695. DLG_V_BLANK_END, &dlg_attr->dlg_vblank_end);
  696. REG_GET(BLANK_OFFSET_1,
  697. MIN_DST_Y_NEXT_START, &dlg_attr->min_dst_y_next_start);
  698. REG_GET(DST_DIMENSIONS,
  699. REFCYC_PER_HTOTAL, &dlg_attr->refcyc_per_htotal);
  700. REG_GET_2(DST_AFTER_SCALER,
  701. REFCYC_X_AFTER_SCALER, &dlg_attr->refcyc_x_after_scaler,
  702. DST_Y_AFTER_SCALER, &dlg_attr->dst_y_after_scaler);
  703. if (REG(PREFETCH_SETTINS))
  704. REG_GET_2(PREFETCH_SETTINS,
  705. DST_Y_PREFETCH, &dlg_attr->dst_y_prefetch,
  706. VRATIO_PREFETCH, &dlg_attr->vratio_prefetch);
  707. else
  708. REG_GET_2(PREFETCH_SETTINGS,
  709. DST_Y_PREFETCH, &dlg_attr->dst_y_prefetch,
  710. VRATIO_PREFETCH, &dlg_attr->vratio_prefetch);
  711. REG_GET_2(VBLANK_PARAMETERS_0,
  712. DST_Y_PER_VM_VBLANK, &dlg_attr->dst_y_per_vm_vblank,
  713. DST_Y_PER_ROW_VBLANK, &dlg_attr->dst_y_per_row_vblank);
  714. REG_GET(REF_FREQ_TO_PIX_FREQ,
  715. REF_FREQ_TO_PIX_FREQ, &dlg_attr->ref_freq_to_pix_freq);
  716. /* DLG - Per luma/chroma */
  717. REG_GET(VBLANK_PARAMETERS_1,
  718. REFCYC_PER_PTE_GROUP_VBLANK_L, &dlg_attr->refcyc_per_pte_group_vblank_l);
  719. REG_GET(VBLANK_PARAMETERS_3,
  720. REFCYC_PER_META_CHUNK_VBLANK_L, &dlg_attr->refcyc_per_meta_chunk_vblank_l);
  721. if (REG(NOM_PARAMETERS_0))
  722. REG_GET(NOM_PARAMETERS_0,
  723. DST_Y_PER_PTE_ROW_NOM_L, &dlg_attr->dst_y_per_pte_row_nom_l);
  724. if (REG(NOM_PARAMETERS_1))
  725. REG_GET(NOM_PARAMETERS_1,
  726. REFCYC_PER_PTE_GROUP_NOM_L, &dlg_attr->refcyc_per_pte_group_nom_l);
  727. REG_GET(NOM_PARAMETERS_4,
  728. DST_Y_PER_META_ROW_NOM_L, &dlg_attr->dst_y_per_meta_row_nom_l);
  729. REG_GET(NOM_PARAMETERS_5,
  730. REFCYC_PER_META_CHUNK_NOM_L, &dlg_attr->refcyc_per_meta_chunk_nom_l);
  731. REG_GET_2(PER_LINE_DELIVERY_PRE,
  732. REFCYC_PER_LINE_DELIVERY_PRE_L, &dlg_attr->refcyc_per_line_delivery_pre_l,
  733. REFCYC_PER_LINE_DELIVERY_PRE_C, &dlg_attr->refcyc_per_line_delivery_pre_c);
  734. REG_GET_2(PER_LINE_DELIVERY,
  735. REFCYC_PER_LINE_DELIVERY_L, &dlg_attr->refcyc_per_line_delivery_l,
  736. REFCYC_PER_LINE_DELIVERY_C, &dlg_attr->refcyc_per_line_delivery_c);
  737. if (REG(PREFETCH_SETTINS_C))
  738. REG_GET(PREFETCH_SETTINS_C,
  739. VRATIO_PREFETCH_C, &dlg_attr->vratio_prefetch_c);
  740. else
  741. REG_GET(PREFETCH_SETTINGS_C,
  742. VRATIO_PREFETCH_C, &dlg_attr->vratio_prefetch_c);
  743. REG_GET(VBLANK_PARAMETERS_2,
  744. REFCYC_PER_PTE_GROUP_VBLANK_C, &dlg_attr->refcyc_per_pte_group_vblank_c);
  745. REG_GET(VBLANK_PARAMETERS_4,
  746. REFCYC_PER_META_CHUNK_VBLANK_C, &dlg_attr->refcyc_per_meta_chunk_vblank_c);
  747. if (REG(NOM_PARAMETERS_2))
  748. REG_GET(NOM_PARAMETERS_2,
  749. DST_Y_PER_PTE_ROW_NOM_C, &dlg_attr->dst_y_per_pte_row_nom_c);
  750. if (REG(NOM_PARAMETERS_3))
  751. REG_GET(NOM_PARAMETERS_3,
  752. REFCYC_PER_PTE_GROUP_NOM_C, &dlg_attr->refcyc_per_pte_group_nom_c);
  753. REG_GET(NOM_PARAMETERS_6,
  754. DST_Y_PER_META_ROW_NOM_C, &dlg_attr->dst_y_per_meta_row_nom_c);
  755. REG_GET(NOM_PARAMETERS_7,
  756. REFCYC_PER_META_CHUNK_NOM_C, &dlg_attr->refcyc_per_meta_chunk_nom_c);
  757. /* TTU - per hubp */
  758. REG_GET_2(DCN_TTU_QOS_WM,
  759. QoS_LEVEL_LOW_WM, &ttu_attr->qos_level_low_wm,
  760. QoS_LEVEL_HIGH_WM, &ttu_attr->qos_level_high_wm);
  761. REG_GET_2(DCN_GLOBAL_TTU_CNTL,
  762. MIN_TTU_VBLANK, &ttu_attr->min_ttu_vblank,
  763. QoS_LEVEL_FLIP, &ttu_attr->qos_level_flip);
  764. /* TTU - per luma/chroma */
  765. /* Assumed surf0 is luma and 1 is chroma */
  766. REG_GET_3(DCN_SURF0_TTU_CNTL0,
  767. REFCYC_PER_REQ_DELIVERY, &ttu_attr->refcyc_per_req_delivery_l,
  768. QoS_LEVEL_FIXED, &ttu_attr->qos_level_fixed_l,
  769. QoS_RAMP_DISABLE, &ttu_attr->qos_ramp_disable_l);
  770. REG_GET(DCN_SURF0_TTU_CNTL1,
  771. REFCYC_PER_REQ_DELIVERY_PRE,
  772. &ttu_attr->refcyc_per_req_delivery_pre_l);
  773. REG_GET_3(DCN_SURF1_TTU_CNTL0,
  774. REFCYC_PER_REQ_DELIVERY, &ttu_attr->refcyc_per_req_delivery_c,
  775. QoS_LEVEL_FIXED, &ttu_attr->qos_level_fixed_c,
  776. QoS_RAMP_DISABLE, &ttu_attr->qos_ramp_disable_c);
  777. REG_GET(DCN_SURF1_TTU_CNTL1,
  778. REFCYC_PER_REQ_DELIVERY_PRE,
  779. &ttu_attr->refcyc_per_req_delivery_pre_c);
  780. /* Rest of hubp */
  781. REG_GET(DCSURF_SURFACE_CONFIG,
  782. SURFACE_PIXEL_FORMAT, &s->pixel_format);
  783. REG_GET(DCSURF_SURFACE_EARLIEST_INUSE_HIGH,
  784. SURFACE_EARLIEST_INUSE_ADDRESS_HIGH, &s->inuse_addr_hi);
  785. REG_GET_2(DCSURF_PRI_VIEWPORT_DIMENSION,
  786. PRI_VIEWPORT_WIDTH, &s->viewport_width,
  787. PRI_VIEWPORT_HEIGHT, &s->viewport_height);
  788. REG_GET_2(DCSURF_SURFACE_CONFIG,
  789. ROTATION_ANGLE, &s->rotation_angle,
  790. H_MIRROR_EN, &s->h_mirror_en);
  791. REG_GET(DCSURF_TILING_CONFIG,
  792. SW_MODE, &s->sw_mode);
  793. REG_GET(DCSURF_SURFACE_CONTROL,
  794. PRIMARY_SURFACE_DCC_EN, &s->dcc_en);
  795. REG_GET_3(DCHUBP_CNTL,
  796. HUBP_BLANK_EN, &s->blank_en,
  797. HUBP_TTU_DISABLE, &s->ttu_disable,
  798. HUBP_UNDERFLOW_STATUS, &s->underflow_status);
  799. REG_GET(DCN_GLOBAL_TTU_CNTL,
  800. MIN_TTU_VBLANK, &s->min_ttu_vblank);
  801. REG_GET_2(DCN_TTU_QOS_WM,
  802. QoS_LEVEL_LOW_WM, &s->qos_level_low_wm,
  803. QoS_LEVEL_HIGH_WM, &s->qos_level_high_wm);
  804. }
  805. enum cursor_pitch hubp1_get_cursor_pitch(unsigned int pitch)
  806. {
  807. enum cursor_pitch hw_pitch;
  808. switch (pitch) {
  809. case 64:
  810. hw_pitch = CURSOR_PITCH_64_PIXELS;
  811. break;
  812. case 128:
  813. hw_pitch = CURSOR_PITCH_128_PIXELS;
  814. break;
  815. case 256:
  816. hw_pitch = CURSOR_PITCH_256_PIXELS;
  817. break;
  818. default:
  819. DC_ERR("Invalid cursor pitch of %d. "
  820. "Only 64/128/256 is supported on DCN.\n", pitch);
  821. hw_pitch = CURSOR_PITCH_64_PIXELS;
  822. break;
  823. }
  824. return hw_pitch;
  825. }
  826. static enum cursor_lines_per_chunk hubp1_get_lines_per_chunk(
  827. unsigned int cur_width,
  828. enum dc_cursor_color_format format)
  829. {
  830. enum cursor_lines_per_chunk line_per_chunk;
  831. if (format == CURSOR_MODE_MONO)
  832. /* impl B. expansion in CUR Buffer reader */
  833. line_per_chunk = CURSOR_LINE_PER_CHUNK_16;
  834. else if (cur_width <= 32)
  835. line_per_chunk = CURSOR_LINE_PER_CHUNK_16;
  836. else if (cur_width <= 64)
  837. line_per_chunk = CURSOR_LINE_PER_CHUNK_8;
  838. else if (cur_width <= 128)
  839. line_per_chunk = CURSOR_LINE_PER_CHUNK_4;
  840. else
  841. line_per_chunk = CURSOR_LINE_PER_CHUNK_2;
  842. return line_per_chunk;
  843. }
  844. void hubp1_cursor_set_attributes(
  845. struct hubp *hubp,
  846. const struct dc_cursor_attributes *attr)
  847. {
  848. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  849. enum cursor_pitch hw_pitch = hubp1_get_cursor_pitch(attr->pitch);
  850. enum cursor_lines_per_chunk lpc = hubp1_get_lines_per_chunk(
  851. attr->width, attr->color_format);
  852. hubp->curs_attr = *attr;
  853. REG_UPDATE(CURSOR_SURFACE_ADDRESS_HIGH,
  854. CURSOR_SURFACE_ADDRESS_HIGH, attr->address.high_part);
  855. REG_UPDATE(CURSOR_SURFACE_ADDRESS,
  856. CURSOR_SURFACE_ADDRESS, attr->address.low_part);
  857. REG_UPDATE_2(CURSOR_SIZE,
  858. CURSOR_WIDTH, attr->width,
  859. CURSOR_HEIGHT, attr->height);
  860. REG_UPDATE_3(CURSOR_CONTROL,
  861. CURSOR_MODE, attr->color_format,
  862. CURSOR_PITCH, hw_pitch,
  863. CURSOR_LINES_PER_CHUNK, lpc);
  864. REG_SET_2(CURSOR_SETTINS, 0,
  865. /* no shift of the cursor HDL schedule */
  866. CURSOR0_DST_Y_OFFSET, 0,
  867. /* used to shift the cursor chunk request deadline */
  868. CURSOR0_CHUNK_HDL_ADJUST, 3);
  869. }
  870. void hubp1_cursor_set_position(
  871. struct hubp *hubp,
  872. const struct dc_cursor_position *pos,
  873. const struct dc_cursor_mi_param *param)
  874. {
  875. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  876. int src_x_offset = pos->x - pos->x_hotspot - param->viewport_x_start;
  877. uint32_t cur_en = pos->enable ? 1 : 0;
  878. uint32_t dst_x_offset = (src_x_offset >= 0) ? src_x_offset : 0;
  879. /*
  880. * Guard aganst cursor_set_position() from being called with invalid
  881. * attributes
  882. *
  883. * TODO: Look at combining cursor_set_position() and
  884. * cursor_set_attributes() into cursor_update()
  885. */
  886. if (hubp->curs_attr.address.quad_part == 0)
  887. return;
  888. dst_x_offset *= param->ref_clk_khz;
  889. dst_x_offset /= param->pixel_clk_khz;
  890. ASSERT(param->h_scale_ratio.value);
  891. if (param->h_scale_ratio.value)
  892. dst_x_offset = dc_fixpt_floor(dc_fixpt_div(
  893. dc_fixpt_from_int(dst_x_offset),
  894. param->h_scale_ratio));
  895. if (src_x_offset >= (int)param->viewport_width)
  896. cur_en = 0; /* not visible beyond right edge*/
  897. if (src_x_offset + (int)hubp->curs_attr.width <= 0)
  898. cur_en = 0; /* not visible beyond left edge*/
  899. if (cur_en && REG_READ(CURSOR_SURFACE_ADDRESS) == 0)
  900. hubp->funcs->set_cursor_attributes(hubp, &hubp->curs_attr);
  901. REG_UPDATE(CURSOR_CONTROL,
  902. CURSOR_ENABLE, cur_en);
  903. REG_SET_2(CURSOR_POSITION, 0,
  904. CURSOR_X_POSITION, pos->x,
  905. CURSOR_Y_POSITION, pos->y);
  906. REG_SET_2(CURSOR_HOT_SPOT, 0,
  907. CURSOR_HOT_SPOT_X, pos->x_hotspot,
  908. CURSOR_HOT_SPOT_Y, pos->y_hotspot);
  909. REG_SET(CURSOR_DST_OFFSET, 0,
  910. CURSOR_DST_X_OFFSET, dst_x_offset);
  911. /* TODO Handle surface pixel formats other than 4:4:4 */
  912. }
  913. void hubp1_clk_cntl(struct hubp *hubp, bool enable)
  914. {
  915. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  916. uint32_t clk_enable = enable ? 1 : 0;
  917. REG_UPDATE(HUBP_CLK_CNTL, HUBP_CLOCK_ENABLE, clk_enable);
  918. }
  919. void hubp1_vtg_sel(struct hubp *hubp, uint32_t otg_inst)
  920. {
  921. struct dcn10_hubp *hubp1 = TO_DCN10_HUBP(hubp);
  922. REG_UPDATE(DCHUBP_CNTL, HUBP_VTG_SEL, otg_inst);
  923. }
  924. static struct hubp_funcs dcn10_hubp_funcs = {
  925. .hubp_program_surface_flip_and_addr =
  926. hubp1_program_surface_flip_and_addr,
  927. .hubp_program_surface_config =
  928. hubp1_program_surface_config,
  929. .hubp_is_flip_pending = hubp1_is_flip_pending,
  930. .hubp_setup = hubp1_setup,
  931. .hubp_set_vm_system_aperture_settings = hubp1_set_vm_system_aperture_settings,
  932. .hubp_set_vm_context0_settings = hubp1_set_vm_context0_settings,
  933. .set_blank = hubp1_set_blank,
  934. .dcc_control = hubp1_dcc_control,
  935. .mem_program_viewport = min_set_viewport,
  936. .set_hubp_blank_en = hubp1_set_hubp_blank_en,
  937. .set_cursor_attributes = hubp1_cursor_set_attributes,
  938. .set_cursor_position = hubp1_cursor_set_position,
  939. .hubp_disconnect = hubp1_disconnect,
  940. .hubp_clk_cntl = hubp1_clk_cntl,
  941. .hubp_vtg_sel = hubp1_vtg_sel,
  942. .hubp_read_state = hubp1_read_state,
  943. .hubp_disable_control = hubp1_disable_control,
  944. .hubp_get_underflow_status = hubp1_get_underflow_status,
  945. };
  946. /*****************************************/
  947. /* Constructor, Destructor */
  948. /*****************************************/
  949. void dcn10_hubp_construct(
  950. struct dcn10_hubp *hubp1,
  951. struct dc_context *ctx,
  952. uint32_t inst,
  953. const struct dcn_mi_registers *hubp_regs,
  954. const struct dcn_mi_shift *hubp_shift,
  955. const struct dcn_mi_mask *hubp_mask)
  956. {
  957. hubp1->base.funcs = &dcn10_hubp_funcs;
  958. hubp1->base.ctx = ctx;
  959. hubp1->hubp_regs = hubp_regs;
  960. hubp1->hubp_shift = hubp_shift;
  961. hubp1->hubp_mask = hubp_mask;
  962. hubp1->base.inst = inst;
  963. hubp1->base.opp_id = 0xf;
  964. hubp1->base.mpcc_id = 0xf;
  965. }