sde_wb.c 26 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2021-2022 Qualcomm Innovation Center, Inc. All rights reserved.
  4. * Copyright (c) 2015-2021, The Linux Foundation. All rights reserved.
  5. */
  6. #define pr_fmt(fmt) "[drm:%s:%d] " fmt, __func__, __LINE__
  7. #include <drm/sde_drm.h>
  8. #include <drm/drm_probe_helper.h>
  9. #include "msm_kms.h"
  10. #include "sde_kms.h"
  11. #include "sde_wb.h"
  12. #include "sde_formats.h"
  13. /* maximum display mode resolution if not available from catalog */
  14. #define SDE_WB_MODE_MAX_WIDTH 5120
  15. #define SDE_WB_MODE_MAX_HEIGHT 5120
  16. static const struct drm_display_mode sde_custom_wb_modes[] = {
  17. /* 5120x2160@60Hz */
  18. { DRM_MODE("5120x2160", DRM_MODE_TYPE_DRIVER, 693264, 5120, 5128,
  19. 5160, 5200, 0, 2160, 2208, 2216, 2222, 0,
  20. DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_NVSYNC) },
  21. { DRM_MODE("2160x5120", DRM_MODE_TYPE_DRIVER, 693264, 2160, 2208,
  22. 2216, 2222, 0, 5120, 5128, 5160, 5200, 0,
  23. DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_NVSYNC) },
  24. { DRM_MODE("5120x2560", DRM_MODE_TYPE_DRIVER, 818064, 5120, 5128,
  25. 5160, 5200, 0, 2560, 2608, 2616, 2622, 0,
  26. DRM_MODE_FLAG_PHSYNC | DRM_MODE_FLAG_NVSYNC) },
  27. };
  28. /* Serialization lock for sde_wb_list */
  29. static DEFINE_MUTEX(sde_wb_list_lock);
  30. /* List of all writeback devices installed */
  31. static LIST_HEAD(sde_wb_list);
  32. /**
  33. * sde_wb_is_format_valid - check if given format/modifier is supported
  34. * @wb_dev: Pointer to writeback device
  35. * @pixel_format: Fourcc pixel format
  36. * @format_modifier: Format modifier
  37. * Returns: true if valid; false otherwise
  38. */
  39. static int sde_wb_is_format_valid(struct sde_wb_device *wb_dev,
  40. u32 pixel_format, u64 format_modifier)
  41. {
  42. const struct sde_format_extended *fmts = wb_dev->wb_cfg->format_list;
  43. int i;
  44. if (!fmts)
  45. return false;
  46. for (i = 0; fmts[i].fourcc_format; i++)
  47. if ((fmts[i].modifier == format_modifier) &&
  48. (fmts[i].fourcc_format == pixel_format))
  49. return true;
  50. return false;
  51. }
  52. enum drm_connector_status
  53. sde_wb_connector_detect(struct drm_connector *connector,
  54. bool force,
  55. void *display)
  56. {
  57. enum drm_connector_status rc = connector_status_unknown;
  58. SDE_DEBUG("\n");
  59. if (display)
  60. rc = ((struct sde_wb_device *)display)->detect_status;
  61. return rc;
  62. }
  63. static int sde_wb_connector_add_custom_modes(struct drm_connector *connector,
  64. u32 hdisplay, u32 vdisplay)
  65. {
  66. int i, num_modes = 0;
  67. struct drm_display_mode *mode;
  68. struct drm_device *dev = connector->dev;
  69. if (!hdisplay || !vdisplay)
  70. return 0;
  71. if (hdisplay > SDE_WB_MODE_MAX_WIDTH)
  72. hdisplay = SDE_WB_MODE_MAX_WIDTH;
  73. if (vdisplay > SDE_WB_MODE_MAX_HEIGHT)
  74. vdisplay = SDE_WB_MODE_MAX_HEIGHT;
  75. for (i = 0; i < ARRAY_SIZE(sde_custom_wb_modes); i++) {
  76. const struct drm_display_mode *ptr = &sde_custom_wb_modes[i];
  77. if (ptr->hdisplay > hdisplay || ptr->vdisplay > vdisplay)
  78. continue;
  79. mode = drm_mode_duplicate(dev, ptr);
  80. if (mode) {
  81. drm_mode_probed_add(connector, mode);
  82. num_modes++;
  83. }
  84. }
  85. return num_modes;
  86. }
  87. int sde_wb_connector_get_modes(struct drm_connector *connector, void *display,
  88. const struct msm_resource_caps_info *avail_res)
  89. {
  90. struct sde_wb_device *wb_dev;
  91. int num_modes = 0;
  92. if (!connector || !display)
  93. return 0;
  94. wb_dev = display;
  95. SDE_DEBUG("\n");
  96. mutex_lock(&wb_dev->wb_lock);
  97. if (wb_dev->count_modes && wb_dev->modes) {
  98. struct drm_display_mode *mode;
  99. int i, ret;
  100. for (i = 0; i < wb_dev->count_modes; i++) {
  101. mode = drm_mode_create(connector->dev);
  102. if (!mode) {
  103. SDE_ERROR("failed to create mode\n");
  104. break;
  105. }
  106. ret = drm_mode_convert_umode(wb_dev->drm_dev, mode,
  107. &wb_dev->modes[i]);
  108. if (ret) {
  109. SDE_ERROR("failed to convert mode %d\n", ret);
  110. break;
  111. }
  112. drm_mode_probed_add(connector, mode);
  113. num_modes++;
  114. }
  115. } else {
  116. u32 max_width = SDE_WB_MODE_MAX_WIDTH;
  117. if (wb_dev->wb_cfg && wb_dev->wb_cfg->sblk)
  118. max_width = max(wb_dev->wb_cfg->sblk->maxlinewidth,
  119. wb_dev->wb_cfg->sblk->maxlinewidth_linear);
  120. num_modes = drm_add_modes_noedid(connector, max_width,
  121. SDE_WB_MODE_MAX_HEIGHT);
  122. num_modes += sde_wb_connector_add_custom_modes(connector, max_width,
  123. SDE_WB_MODE_MAX_HEIGHT);
  124. }
  125. mutex_unlock(&wb_dev->wb_lock);
  126. return num_modes;
  127. }
  128. struct drm_framebuffer *
  129. sde_wb_connector_state_get_output_fb(struct drm_connector_state *state)
  130. {
  131. if (!state || !state->connector ||
  132. (state->connector->connector_type !=
  133. DRM_MODE_CONNECTOR_VIRTUAL)) {
  134. SDE_ERROR("invalid params\n");
  135. return NULL;
  136. }
  137. SDE_DEBUG("\n");
  138. return sde_connector_get_out_fb(state);
  139. }
  140. int sde_wb_connector_state_get_output_roi(struct drm_connector_state *state,
  141. struct sde_rect *roi)
  142. {
  143. if (!state || !roi || !state->connector ||
  144. (state->connector->connector_type !=
  145. DRM_MODE_CONNECTOR_VIRTUAL)) {
  146. SDE_ERROR("invalid params\n");
  147. return -EINVAL;
  148. }
  149. SDE_DEBUG("\n");
  150. roi->x = sde_connector_get_property(state, CONNECTOR_PROP_DST_X);
  151. roi->y = sde_connector_get_property(state, CONNECTOR_PROP_DST_Y);
  152. roi->w = sde_connector_get_property(state, CONNECTOR_PROP_DST_W);
  153. roi->h = sde_connector_get_property(state, CONNECTOR_PROP_DST_H);
  154. return 0;
  155. }
  156. /**
  157. * sde_wb_connector_set_modes - set writeback modes and connection status
  158. * @wb_dev: Pointer to write back device
  159. * @count_modes: Count of modes
  160. * @modes: Pointer to writeback mode requested
  161. * @connected: Connection status requested
  162. * Returns: 0 if success; error code otherwise
  163. */
  164. static
  165. int sde_wb_connector_set_modes(struct sde_wb_device *wb_dev,
  166. u32 count_modes, struct drm_mode_modeinfo __user *modes,
  167. bool connected)
  168. {
  169. struct drm_mode_modeinfo *modeinfo = NULL;
  170. int ret = 0;
  171. int i;
  172. if (!wb_dev || !wb_dev->connector ||
  173. (wb_dev->connector->connector_type !=
  174. DRM_MODE_CONNECTOR_VIRTUAL)) {
  175. SDE_ERROR("invalid params\n");
  176. return -EINVAL;
  177. }
  178. SDE_DEBUG("\n");
  179. if (connected) {
  180. SDE_DEBUG("connect\n");
  181. if (!count_modes || !modes) {
  182. SDE_ERROR("invalid count_modes :%u and modes :%d\n",
  183. count_modes, !modes);
  184. return -EINVAL;
  185. }
  186. modeinfo = kcalloc(count_modes,
  187. sizeof(struct drm_mode_modeinfo),
  188. GFP_KERNEL);
  189. if (!modeinfo) {
  190. SDE_ERROR("invalid params\n");
  191. ret = -ENOMEM;
  192. goto error;
  193. }
  194. if (copy_from_user(modeinfo, modes,
  195. count_modes *
  196. sizeof(struct drm_mode_modeinfo))) {
  197. SDE_ERROR("failed to copy modes\n");
  198. kfree(modeinfo);
  199. ret = -EFAULT;
  200. goto error;
  201. }
  202. for (i = 0; i < count_modes; i++) {
  203. struct drm_display_mode dispmode;
  204. memset(&dispmode, 0, sizeof(dispmode));
  205. ret = drm_mode_convert_umode(wb_dev->drm_dev,
  206. &dispmode, &modeinfo[i]);
  207. if (ret) {
  208. SDE_ERROR(
  209. "failed to convert mode %d:\"%s\" %d %d %d %d %d %d %d %d %d %d 0x%x 0x%x status:%d rc:%d\n",
  210. i,
  211. modeinfo[i].name,
  212. modeinfo[i].vrefresh,
  213. modeinfo[i].clock,
  214. modeinfo[i].hdisplay,
  215. modeinfo[i].hsync_start,
  216. modeinfo[i].hsync_end,
  217. modeinfo[i].htotal,
  218. modeinfo[i].vdisplay,
  219. modeinfo[i].vsync_start,
  220. modeinfo[i].vsync_end,
  221. modeinfo[i].vtotal,
  222. modeinfo[i].type,
  223. modeinfo[i].flags,
  224. dispmode.status,
  225. ret);
  226. kfree(modeinfo);
  227. goto error;
  228. }
  229. }
  230. if (wb_dev->modes) {
  231. wb_dev->count_modes = 0;
  232. kfree(wb_dev->modes);
  233. wb_dev->modes = NULL;
  234. }
  235. wb_dev->count_modes = count_modes;
  236. wb_dev->modes = modeinfo;
  237. wb_dev->detect_status = connector_status_connected;
  238. } else {
  239. SDE_DEBUG("disconnect\n");
  240. if (wb_dev->modes) {
  241. wb_dev->count_modes = 0;
  242. kfree(wb_dev->modes);
  243. wb_dev->modes = NULL;
  244. }
  245. wb_dev->detect_status = connector_status_disconnected;
  246. }
  247. error:
  248. return ret;
  249. }
  250. static void _sde_wb_connector_clear_dnsc_blur(struct drm_connector_state *state)
  251. {
  252. struct sde_connector_state *cstate = to_sde_connector_state(state);
  253. int i;
  254. for (i = 0; i < cstate->dnsc_blur_count; i++)
  255. memset(&cstate->dnsc_blur_cfg[i], 0, sizeof(struct sde_drm_dnsc_blur_cfg));
  256. cstate->dnsc_blur_count = 0;
  257. }
  258. static int _sde_wb_connector_set_dnsc_blur(struct sde_wb_device *wb_dev,
  259. struct drm_connector_state *state, void __user *usr_ptr)
  260. {
  261. struct sde_connector_state *cstate = to_sde_connector_state(state);
  262. struct sde_kms *sde_kms = sde_connector_get_kms(wb_dev->connector);
  263. struct sde_drm_dnsc_blur_cfg *dnsc_blur_cfg = &cstate->dnsc_blur_cfg[0];
  264. u32 copy_count;
  265. int ret = 0, i;
  266. if (!sde_kms || !sde_kms->catalog)
  267. return -EINVAL;
  268. if (!usr_ptr)
  269. goto disable;
  270. /* copy only the first block */
  271. if (copy_from_user(dnsc_blur_cfg, usr_ptr, sizeof(struct sde_drm_dnsc_blur_cfg))) {
  272. SDE_ERROR("failed to copy dnsc_blur block 0 data\n");
  273. ret = -EINVAL;
  274. goto disable;
  275. }
  276. if (dnsc_blur_cfg->num_blocks > sde_kms->catalog->dnsc_blur_count) {
  277. SDE_ERROR("invalid number of dnsc_blur blocks:%d\n", dnsc_blur_cfg->num_blocks);
  278. ret = -EINVAL;
  279. goto disable;
  280. }
  281. /* no further data required */
  282. if (dnsc_blur_cfg->num_blocks <= 1)
  283. goto end;
  284. dnsc_blur_cfg++;
  285. usr_ptr += sizeof(struct sde_drm_dnsc_blur_cfg);
  286. copy_count = dnsc_blur_cfg->num_blocks - 1;
  287. /* copy rest of the blocks */
  288. if ((dnsc_blur_cfg->flags & DNSC_BLUR_INDEPENDENT_BLK_CFG)) {
  289. if (copy_from_user(dnsc_blur_cfg, usr_ptr,
  290. copy_count * sizeof(struct sde_drm_dnsc_blur_cfg))) {
  291. SDE_ERROR("failed to copy dnsc_blur data\n");
  292. ret = -EINVAL;
  293. goto disable;
  294. }
  295. /* duplicate rest of the blocks */
  296. } else if (dnsc_blur_cfg->flags & DNSC_BLUR_MIRROR_BLK_CFG) {
  297. for (i = 0; i < copy_count; i++) {
  298. memcpy(dnsc_blur_cfg, &cstate->dnsc_blur_cfg[0],
  299. sizeof(struct sde_drm_dnsc_blur_cfg));
  300. dnsc_blur_cfg++;
  301. }
  302. }
  303. end:
  304. cstate->dnsc_blur_count = dnsc_blur_cfg->num_blocks;
  305. return 0;
  306. disable:
  307. _sde_wb_connector_clear_dnsc_blur(state);
  308. return ret;
  309. }
  310. static int _sde_wb_connector_set_out_fb(struct sde_wb_device *wb_dev,
  311. struct drm_connector_state *state)
  312. {
  313. struct drm_framebuffer *out_fb;
  314. const struct sde_format *sde_format;
  315. int rc = 0;
  316. out_fb = sde_connector_get_out_fb(state);
  317. if (!out_fb)
  318. goto end;
  319. sde_format = sde_get_sde_format_ext(out_fb->format->format, out_fb->modifier);
  320. if (!sde_format) {
  321. SDE_ERROR("failed to get sde format\n");
  322. rc = -EINVAL;
  323. goto end;
  324. }
  325. if (!sde_wb_is_format_valid(wb_dev, out_fb->format->format, out_fb->modifier)) {
  326. SDE_ERROR("unsupported writeback format 0x%x/0x%llx\n",
  327. out_fb->format->format, out_fb->modifier);
  328. rc = -EINVAL;
  329. goto end;
  330. }
  331. end:
  332. return rc;
  333. }
  334. int sde_wb_connector_set_property(struct drm_connector *connector,
  335. struct drm_connector_state *state, int idx, uint64_t value, void *display)
  336. {
  337. struct sde_wb_device *wb_dev = display;
  338. int rc = 0;
  339. if (!connector || !state || !wb_dev) {
  340. SDE_ERROR("invalid argument(s)\n");
  341. return -EINVAL;
  342. }
  343. switch (idx) {
  344. case CONNECTOR_PROP_OUT_FB:
  345. rc = _sde_wb_connector_set_out_fb(wb_dev, state);
  346. break;
  347. case CONNECTOR_PROP_DNSC_BLUR:
  348. rc = _sde_wb_connector_set_dnsc_blur(wb_dev, state,
  349. (void __user *)(uintptr_t)value);
  350. break;
  351. default:
  352. /* nothing to do */
  353. break;
  354. }
  355. return rc;
  356. }
  357. int sde_wb_get_info(struct drm_connector *connector,
  358. struct msm_display_info *info, void *display)
  359. {
  360. struct sde_wb_device *wb_dev = display;
  361. u32 max_width = SDE_WB_MODE_MAX_WIDTH;
  362. if (!info || !wb_dev) {
  363. pr_err("invalid params\n");
  364. return -EINVAL;
  365. }
  366. if (wb_dev->wb_cfg && wb_dev->wb_cfg->sblk)
  367. max_width = max(wb_dev->wb_cfg->sblk->maxlinewidth,
  368. wb_dev->wb_cfg->sblk->maxlinewidth_linear);
  369. memset(info, 0, sizeof(struct msm_display_info));
  370. info->intf_type = DRM_MODE_CONNECTOR_VIRTUAL;
  371. info->num_of_h_tiles = 1;
  372. info->h_tile_instance[0] = sde_wb_get_index(display);
  373. info->is_connected = true;
  374. info->capabilities = MSM_DISPLAY_CAP_HOT_PLUG | MSM_DISPLAY_CAP_EDID;
  375. info->max_width = max_width;
  376. info->max_height = SDE_WB_MODE_MAX_HEIGHT;
  377. return 0;
  378. }
  379. int sde_wb_get_mode_info(struct drm_connector *connector,
  380. const struct drm_display_mode *drm_mode,
  381. struct msm_sub_mode *sub_mode,
  382. struct msm_mode_info *mode_info,
  383. void *display, const struct msm_resource_caps_info *avail_res)
  384. {
  385. const u32 dual_lm = 2;
  386. const u32 single_lm = 1;
  387. const u32 single_intf = 1;
  388. const u32 no_enc = 0;
  389. struct msm_display_topology *topology;
  390. struct sde_wb_device *wb_dev = display;
  391. u16 hdisplay;
  392. int i;
  393. if (!drm_mode || !mode_info || !avail_res ||
  394. !avail_res->max_mixer_width || !display) {
  395. pr_err("invalid params\n");
  396. return -EINVAL;
  397. }
  398. hdisplay = drm_mode->hdisplay;
  399. /* find maximum display width to support */
  400. for (i = 0; i < wb_dev->count_modes; i++)
  401. hdisplay = max(hdisplay, wb_dev->modes[i].hdisplay);
  402. topology = &mode_info->topology;
  403. topology->num_lm = (avail_res->max_mixer_width <= hdisplay) ?
  404. dual_lm : single_lm;
  405. topology->num_enc = no_enc;
  406. topology->num_intf = single_intf;
  407. mode_info->comp_info.comp_type = MSM_DISPLAY_COMPRESSION_NONE;
  408. mode_info->wide_bus_en = false;
  409. mode_info->comp_info.comp_ratio = MSM_DISPLAY_COMPRESSION_RATIO_NONE;
  410. return 0;
  411. }
  412. int sde_wb_connector_set_info_blob(struct drm_connector *connector,
  413. void *info, void *display, struct msm_mode_info *mode_info)
  414. {
  415. struct sde_wb_device *wb_dev = display;
  416. const struct sde_format_extended *format_list;
  417. struct sde_kms *sde_kms;
  418. struct sde_mdss_cfg *catalog;
  419. int i;
  420. if (!connector || !info || !display || !wb_dev->wb_cfg) {
  421. SDE_ERROR("invalid params\n");
  422. return -EINVAL;
  423. }
  424. sde_kms = sde_connector_get_kms(connector);
  425. if (!sde_kms)
  426. return -EINVAL;
  427. catalog = sde_kms->catalog;
  428. format_list = wb_dev->wb_cfg->format_list;
  429. /* Populate info buffer */
  430. if (format_list) {
  431. sde_kms_info_start(info, "pixel_formats");
  432. while (format_list->fourcc_format) {
  433. sde_kms_info_append_format(info, format_list->fourcc_format,
  434. format_list->modifier);
  435. ++format_list;
  436. }
  437. sde_kms_info_stop(info);
  438. }
  439. sde_kms_info_add_keyint(info, "wb_intf_index", wb_dev->wb_idx - WB_0);
  440. sde_kms_info_add_keyint(info, "maxlinewidth", wb_dev->wb_cfg->sblk->maxlinewidth);
  441. sde_kms_info_add_keyint(info, "maxlinewidth_linear",
  442. wb_dev->wb_cfg->sblk->maxlinewidth_linear);
  443. sde_kms_info_start(info, "features");
  444. if (wb_dev->wb_cfg && (wb_dev->wb_cfg->features & BIT(SDE_WB_UBWC)))
  445. sde_kms_info_append(info, "wb_ubwc");
  446. sde_kms_info_stop(info);
  447. sde_kms_info_add_keyint(info, "has_cwb_dither", test_bit(SDE_FEATURE_CWB_DITHER,
  448. catalog->features));
  449. if (catalog->dnsc_blur_count && catalog->dnsc_blur_filters) {
  450. sde_kms_info_add_keyint(info, "dnsc_blur_count", catalog->dnsc_blur_count);
  451. sde_kms_info_start(info, "dnsc_blur_info");
  452. for (i = 0; i < catalog->dnsc_blur_filter_count; i++)
  453. sde_kms_info_append_dnsc_blur_filter_info(info,
  454. &catalog->dnsc_blur_filters[i]);
  455. sde_kms_info_stop(info);
  456. }
  457. return 0;
  458. }
  459. static void _sde_wb_connector_install_dither_property(struct sde_wb_device *wb_dev)
  460. {
  461. struct sde_connector *c_conn = to_sde_connector(wb_dev->connector);
  462. struct sde_kms *sde_kms = sde_connector_get_kms(wb_dev->connector);
  463. struct sde_mdss_cfg *catalog;
  464. char prop_name[DRM_PROP_NAME_LEN];
  465. u32 version = 0;
  466. if (!sde_kms || !sde_kms->catalog)
  467. return;
  468. catalog = sde_kms->catalog;
  469. if (!test_bit(SDE_FEATURE_CWB_DITHER, catalog->features))
  470. return;
  471. version = SDE_COLOR_PROCESS_MAJOR(catalog->pingpong[0].sblk->dither.version);
  472. snprintf(prop_name, ARRAY_SIZE(prop_name), "%s%d", "SDE_PP_CWB_DITHER_V", version);
  473. switch (version) {
  474. case 2:
  475. msm_property_install_blob(&c_conn->property_info, prop_name,
  476. DRM_MODE_PROP_BLOB, CONNECTOR_PROP_PP_CWB_DITHER);
  477. break;
  478. default:
  479. SDE_ERROR("unsupported cwb dither version %d\n", version);
  480. return;
  481. }
  482. }
  483. int sde_wb_connector_post_init(struct drm_connector *connector, void *display)
  484. {
  485. struct sde_connector *c_conn;
  486. struct sde_wb_device *wb_dev = display;
  487. struct msm_drm_private *priv;
  488. struct sde_kms *sde_kms;
  489. struct sde_mdss_cfg *catalog;
  490. static const struct drm_prop_enum_list e_fb_translation_mode[] = {
  491. {SDE_DRM_FB_NON_SEC, "non_sec"},
  492. {SDE_DRM_FB_SEC, "sec"},
  493. };
  494. static const struct drm_prop_enum_list e_cache_state[] = {
  495. {CACHE_STATE_DISABLED, "cache_state_disabled"},
  496. {CACHE_STATE_ENABLED, "cache_state_enabled"},
  497. };
  498. static const struct drm_prop_enum_list e_wb_usage_type[] = {
  499. {WB_USAGE_WFD, "wb_usage_wfd"},
  500. {WB_USAGE_CWB, "wb_usage_cwb"},
  501. {WB_USAGE_OFFLINE_WB, "wb_usage_offline_wb"},
  502. };
  503. if (!connector || !display || !wb_dev->wb_cfg || !wb_dev->drm_dev->dev_private) {
  504. SDE_ERROR("invalid params\n");
  505. return -EINVAL;
  506. }
  507. priv = wb_dev->drm_dev->dev_private;
  508. sde_kms = to_sde_kms(priv->kms);
  509. if (!sde_kms || !sde_kms->catalog) {
  510. SDE_ERROR("invalid sde_kms\n");
  511. return -EINVAL;
  512. }
  513. catalog = sde_kms->catalog;
  514. c_conn = to_sde_connector(connector);
  515. wb_dev->connector = connector;
  516. wb_dev->detect_status = connector_status_connected;
  517. if (test_bit(SDE_SYS_CACHE_DISP, catalog->sde_sys_cache_type_map)
  518. || test_bit(SDE_SYS_CACHE_DISP_WB, catalog->sde_sys_cache_type_map))
  519. msm_property_install_enum(&c_conn->property_info, "cache_state",
  520. 0x0, 0, e_cache_state, ARRAY_SIZE(e_cache_state),
  521. 0, CONNECTOR_PROP_CACHE_STATE);
  522. /*
  523. * Add extra connector properties
  524. */
  525. msm_property_install_range(&c_conn->property_info, "FB_ID",
  526. 0x0, 0, ~0, 0, CONNECTOR_PROP_OUT_FB);
  527. msm_property_install_range(&c_conn->property_info, "DST_X",
  528. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_X);
  529. msm_property_install_range(&c_conn->property_info, "DST_Y",
  530. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_Y);
  531. msm_property_install_range(&c_conn->property_info, "DST_W",
  532. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_W);
  533. msm_property_install_range(&c_conn->property_info, "DST_H",
  534. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_H);
  535. msm_property_install_enum(&c_conn->property_info,
  536. "fb_translation_mode",
  537. 0x0,
  538. 0, e_fb_translation_mode,
  539. ARRAY_SIZE(e_fb_translation_mode), 0,
  540. CONNECTOR_PROP_FB_TRANSLATION_MODE);
  541. if (wb_dev->wb_cfg->features & BIT(SDE_WB_PROG_LINE))
  542. msm_property_install_range(&c_conn->property_info, "early_fence_line",
  543. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_EARLY_FENCE_LINE);
  544. if (catalog->dnsc_blur_count && catalog->dnsc_blur_filters)
  545. msm_property_install_range(&c_conn->property_info, "dnsc_blur",
  546. 0x0, 0, ~0, 0, CONNECTOR_PROP_DNSC_BLUR);
  547. msm_property_install_enum(&c_conn->property_info, "wb_usage_type",
  548. 0x0, 0, e_wb_usage_type, ARRAY_SIZE(e_wb_usage_type),
  549. 0, CONNECTOR_PROP_WB_USAGE_TYPE);
  550. _sde_wb_connector_install_dither_property(wb_dev);
  551. return 0;
  552. }
  553. struct drm_framebuffer *sde_wb_get_output_fb(struct sde_wb_device *wb_dev)
  554. {
  555. struct drm_framebuffer *fb;
  556. if (!wb_dev || !wb_dev->connector) {
  557. SDE_ERROR("invalid params\n");
  558. return NULL;
  559. }
  560. SDE_DEBUG("\n");
  561. mutex_lock(&wb_dev->wb_lock);
  562. fb = sde_wb_connector_state_get_output_fb(wb_dev->connector->state);
  563. mutex_unlock(&wb_dev->wb_lock);
  564. return fb;
  565. }
  566. int sde_wb_get_output_roi(struct sde_wb_device *wb_dev, struct sde_rect *roi)
  567. {
  568. int rc;
  569. if (!wb_dev || !wb_dev->connector || !roi) {
  570. SDE_ERROR("invalid params\n");
  571. return -EINVAL;
  572. }
  573. SDE_DEBUG("\n");
  574. mutex_lock(&wb_dev->wb_lock);
  575. rc = sde_wb_connector_state_get_output_roi(
  576. wb_dev->connector->state, roi);
  577. mutex_unlock(&wb_dev->wb_lock);
  578. return rc;
  579. }
  580. u32 sde_wb_get_num_of_displays(void)
  581. {
  582. u32 count = 0;
  583. struct sde_wb_device *wb_dev;
  584. SDE_DEBUG("\n");
  585. mutex_lock(&sde_wb_list_lock);
  586. list_for_each_entry(wb_dev, &sde_wb_list, wb_list) {
  587. count++;
  588. }
  589. mutex_unlock(&sde_wb_list_lock);
  590. return count;
  591. }
  592. int wb_display_get_displays(void **display_array, u32 max_display_count)
  593. {
  594. struct sde_wb_device *curr;
  595. int i = 0;
  596. SDE_DEBUG("\n");
  597. if (!display_array || !max_display_count) {
  598. if (!display_array)
  599. SDE_ERROR("invalid param\n");
  600. return 0;
  601. }
  602. mutex_lock(&sde_wb_list_lock);
  603. list_for_each_entry(curr, &sde_wb_list, wb_list) {
  604. if (i >= max_display_count)
  605. break;
  606. display_array[i++] = curr;
  607. }
  608. mutex_unlock(&sde_wb_list_lock);
  609. return i;
  610. }
  611. int sde_wb_config(struct drm_device *drm_dev, void *data,
  612. struct drm_file *file_priv)
  613. {
  614. struct sde_drm_wb_cfg *config = data;
  615. struct msm_drm_private *priv;
  616. struct sde_wb_device *wb_dev = NULL;
  617. struct sde_wb_device *curr;
  618. struct drm_connector *connector;
  619. uint32_t flags;
  620. uint32_t connector_id;
  621. uint32_t count_modes;
  622. uint64_t modes;
  623. int rc;
  624. if (!drm_dev || !data) {
  625. SDE_ERROR("invalid params\n");
  626. return -EINVAL;
  627. }
  628. SDE_DEBUG("\n");
  629. flags = config->flags;
  630. connector_id = config->connector_id;
  631. count_modes = config->count_modes;
  632. modes = config->modes;
  633. priv = drm_dev->dev_private;
  634. connector = drm_connector_lookup(drm_dev, file_priv, connector_id);
  635. if (!connector) {
  636. SDE_ERROR("failed to find connector\n");
  637. rc = -ENOENT;
  638. goto fail;
  639. }
  640. mutex_lock(&sde_wb_list_lock);
  641. list_for_each_entry(curr, &sde_wb_list, wb_list) {
  642. if (curr->connector == connector) {
  643. wb_dev = curr;
  644. break;
  645. }
  646. }
  647. mutex_unlock(&sde_wb_list_lock);
  648. if (!wb_dev) {
  649. SDE_ERROR("failed to find wb device\n");
  650. rc = -ENOENT;
  651. goto fail;
  652. }
  653. mutex_lock(&wb_dev->wb_lock);
  654. rc = sde_wb_connector_set_modes(wb_dev, count_modes,
  655. (struct drm_mode_modeinfo __user *) (uintptr_t) modes,
  656. (flags & SDE_DRM_WB_CFG_FLAGS_CONNECTED) ? true : false);
  657. mutex_unlock(&wb_dev->wb_lock);
  658. drm_helper_hpd_irq_event(drm_dev);
  659. fail:
  660. return rc;
  661. }
  662. /**
  663. * _sde_wb_dev_init - perform device initialization
  664. * @wb_dev: Pointer to writeback device
  665. */
  666. static int _sde_wb_dev_init(struct sde_wb_device *wb_dev)
  667. {
  668. int rc = 0;
  669. if (!wb_dev) {
  670. SDE_ERROR("invalid params\n");
  671. return -EINVAL;
  672. }
  673. SDE_DEBUG("\n");
  674. return rc;
  675. }
  676. /**
  677. * _sde_wb_dev_deinit - perform device de-initialization
  678. * @wb_dev: Pointer to writeback device
  679. */
  680. static int _sde_wb_dev_deinit(struct sde_wb_device *wb_dev)
  681. {
  682. int rc = 0;
  683. if (!wb_dev) {
  684. SDE_ERROR("invalid params\n");
  685. return -EINVAL;
  686. }
  687. SDE_DEBUG("\n");
  688. return rc;
  689. }
  690. /**
  691. * sde_wb_bind - bind writeback device with controlling device
  692. * @dev: Pointer to base of platform device
  693. * @master: Pointer to container of drm device
  694. * @data: Pointer to private data
  695. * Returns: Zero on success
  696. */
  697. static int sde_wb_bind(struct device *dev, struct device *master, void *data)
  698. {
  699. struct sde_wb_device *wb_dev;
  700. if (!dev || !master) {
  701. SDE_ERROR("invalid params\n");
  702. return -EINVAL;
  703. }
  704. wb_dev = platform_get_drvdata(to_platform_device(dev));
  705. if (!wb_dev) {
  706. SDE_ERROR("invalid wb device\n");
  707. return -EINVAL;
  708. }
  709. SDE_DEBUG("\n");
  710. mutex_lock(&wb_dev->wb_lock);
  711. wb_dev->drm_dev = dev_get_drvdata(master);
  712. mutex_unlock(&wb_dev->wb_lock);
  713. return 0;
  714. }
  715. /**
  716. * sde_wb_unbind - unbind writeback from controlling device
  717. * @dev: Pointer to base of platform device
  718. * @master: Pointer to container of drm device
  719. * @data: Pointer to private data
  720. */
  721. static void sde_wb_unbind(struct device *dev,
  722. struct device *master, void *data)
  723. {
  724. struct sde_wb_device *wb_dev;
  725. if (!dev) {
  726. SDE_ERROR("invalid params\n");
  727. return;
  728. }
  729. wb_dev = platform_get_drvdata(to_platform_device(dev));
  730. if (!wb_dev) {
  731. SDE_ERROR("invalid wb device\n");
  732. return;
  733. }
  734. SDE_DEBUG("\n");
  735. mutex_lock(&wb_dev->wb_lock);
  736. wb_dev->drm_dev = NULL;
  737. mutex_unlock(&wb_dev->wb_lock);
  738. }
  739. static const struct component_ops sde_wb_comp_ops = {
  740. .bind = sde_wb_bind,
  741. .unbind = sde_wb_unbind,
  742. };
  743. /**
  744. * sde_wb_drm_init - perform DRM initialization
  745. * @wb_dev: Pointer to writeback device
  746. * @encoder: Pointer to associated encoder
  747. */
  748. int sde_wb_drm_init(struct sde_wb_device *wb_dev, struct drm_encoder *encoder)
  749. {
  750. int rc = 0;
  751. if (!wb_dev || !wb_dev->drm_dev || !encoder) {
  752. SDE_ERROR("invalid params\n");
  753. return -EINVAL;
  754. }
  755. SDE_DEBUG("\n");
  756. mutex_lock(&wb_dev->wb_lock);
  757. if (wb_dev->drm_dev->dev_private) {
  758. struct msm_drm_private *priv = wb_dev->drm_dev->dev_private;
  759. struct sde_kms *sde_kms = to_sde_kms(priv->kms);
  760. if (wb_dev->index < sde_kms->catalog->wb_count) {
  761. wb_dev->wb_idx = sde_kms->catalog->wb[wb_dev->index].id;
  762. wb_dev->wb_cfg = &sde_kms->catalog->wb[wb_dev->index];
  763. }
  764. }
  765. wb_dev->drm_dev = encoder->dev;
  766. wb_dev->encoder = encoder;
  767. mutex_unlock(&wb_dev->wb_lock);
  768. return rc;
  769. }
  770. int sde_wb_drm_deinit(struct sde_wb_device *wb_dev)
  771. {
  772. int rc = 0;
  773. if (!wb_dev) {
  774. SDE_ERROR("invalid params\n");
  775. return -EINVAL;
  776. }
  777. SDE_DEBUG("\n");
  778. return rc;
  779. }
  780. /**
  781. * sde_wb_probe - load writeback module
  782. * @pdev: Pointer to platform device
  783. */
  784. static int sde_wb_probe(struct platform_device *pdev)
  785. {
  786. struct sde_wb_device *wb_dev;
  787. int ret;
  788. wb_dev = devm_kzalloc(&pdev->dev, sizeof(*wb_dev), GFP_KERNEL);
  789. if (!wb_dev)
  790. return -ENOMEM;
  791. SDE_DEBUG("\n");
  792. ret = of_property_read_u32(pdev->dev.of_node, "cell-index",
  793. &wb_dev->index);
  794. if (ret) {
  795. SDE_DEBUG("cell index not set, default to 0\n");
  796. wb_dev->index = 0;
  797. }
  798. wb_dev->name = of_get_property(pdev->dev.of_node, "label", NULL);
  799. if (!wb_dev->name) {
  800. SDE_DEBUG("label not set, default to unknown\n");
  801. wb_dev->name = "unknown";
  802. }
  803. wb_dev->wb_idx = SDE_NONE;
  804. mutex_init(&wb_dev->wb_lock);
  805. platform_set_drvdata(pdev, wb_dev);
  806. mutex_lock(&sde_wb_list_lock);
  807. list_add_tail(&wb_dev->wb_list, &sde_wb_list);
  808. mutex_unlock(&sde_wb_list_lock);
  809. if (!_sde_wb_dev_init(wb_dev)) {
  810. ret = component_add(&pdev->dev, &sde_wb_comp_ops);
  811. if (ret)
  812. pr_err("component add failed\n");
  813. }
  814. return ret;
  815. }
  816. /**
  817. * sde_wb_remove - unload writeback module
  818. * @pdev: Pointer to platform device
  819. */
  820. static int sde_wb_remove(struct platform_device *pdev)
  821. {
  822. struct sde_wb_device *wb_dev;
  823. struct sde_wb_device *curr, *next;
  824. wb_dev = platform_get_drvdata(pdev);
  825. if (!wb_dev)
  826. return 0;
  827. SDE_DEBUG("\n");
  828. (void)_sde_wb_dev_deinit(wb_dev);
  829. mutex_lock(&sde_wb_list_lock);
  830. list_for_each_entry_safe(curr, next, &sde_wb_list, wb_list) {
  831. if (curr == wb_dev) {
  832. list_del(&wb_dev->wb_list);
  833. break;
  834. }
  835. }
  836. mutex_unlock(&sde_wb_list_lock);
  837. kfree(wb_dev->modes);
  838. mutex_destroy(&wb_dev->wb_lock);
  839. platform_set_drvdata(pdev, NULL);
  840. devm_kfree(&pdev->dev, wb_dev);
  841. return 0;
  842. }
  843. static const struct of_device_id dt_match[] = {
  844. { .compatible = "qcom,wb-display"},
  845. {}
  846. };
  847. static struct platform_driver sde_wb_driver = {
  848. .probe = sde_wb_probe,
  849. .remove = sde_wb_remove,
  850. .driver = {
  851. .name = "sde_wb",
  852. .of_match_table = dt_match,
  853. .suppress_bind_attrs = true,
  854. },
  855. };
  856. void __init sde_wb_register(void)
  857. {
  858. platform_driver_register(&sde_wb_driver);
  859. }
  860. void __exit sde_wb_unregister(void)
  861. {
  862. platform_driver_unregister(&sde_wb_driver);
  863. }