sde_wb.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2015-2019, The Linux Foundation. All rights reserved.
  4. */
  5. #define pr_fmt(fmt) "[drm:%s:%d] " fmt, __func__, __LINE__
  6. #include <uapi/drm/sde_drm.h>
  7. #include "msm_kms.h"
  8. #include "sde_kms.h"
  9. #include "sde_wb.h"
  10. #include "sde_formats.h"
  11. /* maximum display mode resolution if not available from catalog */
  12. #define SDE_WB_MODE_MAX_WIDTH 4096
  13. #define SDE_WB_MODE_MAX_HEIGHT 4096
  14. /* Serialization lock for sde_wb_list */
  15. static DEFINE_MUTEX(sde_wb_list_lock);
  16. /* List of all writeback devices installed */
  17. static LIST_HEAD(sde_wb_list);
  18. /**
  19. * sde_wb_is_format_valid - check if given format/modifier is supported
  20. * @wb_dev: Pointer to writeback device
  21. * @pixel_format: Fourcc pixel format
  22. * @format_modifier: Format modifier
  23. * Returns: true if valid; false otherwise
  24. */
  25. static int sde_wb_is_format_valid(struct sde_wb_device *wb_dev,
  26. u32 pixel_format, u64 format_modifier)
  27. {
  28. const struct sde_format_extended *fmts = wb_dev->wb_cfg->format_list;
  29. int i;
  30. if (!fmts)
  31. return false;
  32. for (i = 0; fmts[i].fourcc_format; i++)
  33. if ((fmts[i].modifier == format_modifier) &&
  34. (fmts[i].fourcc_format == pixel_format))
  35. return true;
  36. return false;
  37. }
  38. enum drm_connector_status
  39. sde_wb_connector_detect(struct drm_connector *connector,
  40. bool force,
  41. void *display)
  42. {
  43. enum drm_connector_status rc = connector_status_unknown;
  44. SDE_DEBUG("\n");
  45. if (display)
  46. rc = ((struct sde_wb_device *)display)->detect_status;
  47. return rc;
  48. }
  49. int sde_wb_connector_get_modes(struct drm_connector *connector, void *display)
  50. {
  51. struct sde_wb_device *wb_dev;
  52. int num_modes = 0;
  53. if (!connector || !display)
  54. return 0;
  55. wb_dev = display;
  56. SDE_DEBUG("\n");
  57. mutex_lock(&wb_dev->wb_lock);
  58. if (wb_dev->count_modes && wb_dev->modes) {
  59. struct drm_display_mode *mode;
  60. int i, ret;
  61. for (i = 0; i < wb_dev->count_modes; i++) {
  62. mode = drm_mode_create(connector->dev);
  63. if (!mode) {
  64. SDE_ERROR("failed to create mode\n");
  65. break;
  66. }
  67. ret = drm_mode_convert_umode(wb_dev->drm_dev, mode,
  68. &wb_dev->modes[i]);
  69. if (ret) {
  70. SDE_ERROR("failed to convert mode %d\n", ret);
  71. break;
  72. }
  73. drm_mode_probed_add(connector, mode);
  74. num_modes++;
  75. }
  76. } else {
  77. u32 max_width = (wb_dev->wb_cfg && wb_dev->wb_cfg->sblk) ?
  78. wb_dev->wb_cfg->sblk->maxlinewidth :
  79. SDE_WB_MODE_MAX_WIDTH;
  80. num_modes = drm_add_modes_noedid(connector, max_width,
  81. SDE_WB_MODE_MAX_HEIGHT);
  82. }
  83. mutex_unlock(&wb_dev->wb_lock);
  84. return num_modes;
  85. }
  86. struct drm_framebuffer *
  87. sde_wb_connector_state_get_output_fb(struct drm_connector_state *state)
  88. {
  89. if (!state || !state->connector ||
  90. (state->connector->connector_type !=
  91. DRM_MODE_CONNECTOR_VIRTUAL)) {
  92. SDE_ERROR("invalid params\n");
  93. return NULL;
  94. }
  95. SDE_DEBUG("\n");
  96. return sde_connector_get_out_fb(state);
  97. }
  98. int sde_wb_connector_state_get_output_roi(struct drm_connector_state *state,
  99. struct sde_rect *roi)
  100. {
  101. if (!state || !roi || !state->connector ||
  102. (state->connector->connector_type !=
  103. DRM_MODE_CONNECTOR_VIRTUAL)) {
  104. SDE_ERROR("invalid params\n");
  105. return -EINVAL;
  106. }
  107. SDE_DEBUG("\n");
  108. roi->x = sde_connector_get_property(state, CONNECTOR_PROP_DST_X);
  109. roi->y = sde_connector_get_property(state, CONNECTOR_PROP_DST_Y);
  110. roi->w = sde_connector_get_property(state, CONNECTOR_PROP_DST_W);
  111. roi->h = sde_connector_get_property(state, CONNECTOR_PROP_DST_H);
  112. return 0;
  113. }
  114. /**
  115. * sde_wb_connector_set_modes - set writeback modes and connection status
  116. * @wb_dev: Pointer to write back device
  117. * @count_modes: Count of modes
  118. * @modes: Pointer to writeback mode requested
  119. * @connected: Connection status requested
  120. * Returns: 0 if success; error code otherwise
  121. */
  122. static
  123. int sde_wb_connector_set_modes(struct sde_wb_device *wb_dev,
  124. u32 count_modes, struct drm_mode_modeinfo __user *modes,
  125. bool connected)
  126. {
  127. struct drm_mode_modeinfo *modeinfo = NULL;
  128. int ret = 0;
  129. int i;
  130. if (!wb_dev || !wb_dev->connector ||
  131. (wb_dev->connector->connector_type !=
  132. DRM_MODE_CONNECTOR_VIRTUAL)) {
  133. SDE_ERROR("invalid params\n");
  134. return -EINVAL;
  135. }
  136. SDE_DEBUG("\n");
  137. if (connected) {
  138. SDE_DEBUG("connect\n");
  139. if (!count_modes || !modes) {
  140. SDE_ERROR("invalid count_modes :%u and modes :%d\n",
  141. count_modes, !modes);
  142. return -EINVAL;
  143. }
  144. modeinfo = kcalloc(count_modes,
  145. sizeof(struct drm_mode_modeinfo),
  146. GFP_KERNEL);
  147. if (!modeinfo) {
  148. SDE_ERROR("invalid params\n");
  149. ret = -ENOMEM;
  150. goto error;
  151. }
  152. if (copy_from_user(modeinfo, modes,
  153. count_modes *
  154. sizeof(struct drm_mode_modeinfo))) {
  155. SDE_ERROR("failed to copy modes\n");
  156. kfree(modeinfo);
  157. ret = -EFAULT;
  158. goto error;
  159. }
  160. for (i = 0; i < count_modes; i++) {
  161. struct drm_display_mode dispmode;
  162. memset(&dispmode, 0, sizeof(dispmode));
  163. ret = drm_mode_convert_umode(wb_dev->drm_dev,
  164. &dispmode, &modeinfo[i]);
  165. if (ret) {
  166. SDE_ERROR(
  167. "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",
  168. i,
  169. modeinfo[i].name,
  170. modeinfo[i].vrefresh,
  171. modeinfo[i].clock,
  172. modeinfo[i].hdisplay,
  173. modeinfo[i].hsync_start,
  174. modeinfo[i].hsync_end,
  175. modeinfo[i].htotal,
  176. modeinfo[i].vdisplay,
  177. modeinfo[i].vsync_start,
  178. modeinfo[i].vsync_end,
  179. modeinfo[i].vtotal,
  180. modeinfo[i].type,
  181. modeinfo[i].flags,
  182. dispmode.status,
  183. ret);
  184. kfree(modeinfo);
  185. goto error;
  186. }
  187. }
  188. if (wb_dev->modes) {
  189. wb_dev->count_modes = 0;
  190. kfree(wb_dev->modes);
  191. wb_dev->modes = NULL;
  192. }
  193. wb_dev->count_modes = count_modes;
  194. wb_dev->modes = modeinfo;
  195. wb_dev->detect_status = connector_status_connected;
  196. } else {
  197. SDE_DEBUG("disconnect\n");
  198. if (wb_dev->modes) {
  199. wb_dev->count_modes = 0;
  200. kfree(wb_dev->modes);
  201. wb_dev->modes = NULL;
  202. }
  203. wb_dev->detect_status = connector_status_disconnected;
  204. }
  205. error:
  206. return ret;
  207. }
  208. int sde_wb_connector_set_property(struct drm_connector *connector,
  209. struct drm_connector_state *state,
  210. int property_index,
  211. uint64_t value,
  212. void *display)
  213. {
  214. struct sde_wb_device *wb_dev = display;
  215. struct drm_framebuffer *out_fb;
  216. int rc = 0;
  217. SDE_DEBUG("\n");
  218. if (state && (property_index == CONNECTOR_PROP_OUT_FB)) {
  219. const struct sde_format *sde_format;
  220. out_fb = sde_connector_get_out_fb(state);
  221. if (!out_fb)
  222. goto done;
  223. sde_format = sde_get_sde_format_ext(out_fb->format->format,
  224. out_fb->modifier);
  225. if (!sde_format) {
  226. SDE_ERROR("failed to get sde format\n");
  227. rc = -EINVAL;
  228. goto done;
  229. }
  230. if (!sde_wb_is_format_valid(wb_dev, out_fb->format->format,
  231. out_fb->modifier)) {
  232. SDE_ERROR("unsupported writeback format 0x%x/0x%llx\n",
  233. out_fb->format->format,
  234. out_fb->modifier);
  235. rc = -EINVAL;
  236. goto done;
  237. }
  238. }
  239. done:
  240. return rc;
  241. }
  242. int sde_wb_get_info(struct drm_connector *connector,
  243. struct msm_display_info *info, void *display)
  244. {
  245. struct sde_wb_device *wb_dev = display;
  246. if (!info || !wb_dev) {
  247. pr_err("invalid params\n");
  248. return -EINVAL;
  249. }
  250. memset(info, 0, sizeof(struct msm_display_info));
  251. info->intf_type = DRM_MODE_CONNECTOR_VIRTUAL;
  252. info->num_of_h_tiles = 1;
  253. info->h_tile_instance[0] = sde_wb_get_index(display);
  254. info->is_connected = true;
  255. info->capabilities = MSM_DISPLAY_CAP_HOT_PLUG | MSM_DISPLAY_CAP_EDID;
  256. info->max_width = (wb_dev->wb_cfg && wb_dev->wb_cfg->sblk) ?
  257. wb_dev->wb_cfg->sblk->maxlinewidth :
  258. SDE_WB_MODE_MAX_WIDTH;
  259. info->max_height = SDE_WB_MODE_MAX_HEIGHT;
  260. return 0;
  261. }
  262. int sde_wb_get_mode_info(struct drm_connector *connector,
  263. const struct drm_display_mode *drm_mode,
  264. struct msm_mode_info *mode_info,
  265. u32 max_mixer_width, void *display)
  266. {
  267. const u32 dual_lm = 2;
  268. const u32 single_lm = 1;
  269. const u32 single_intf = 1;
  270. const u32 no_enc = 0;
  271. struct msm_display_topology *topology;
  272. struct sde_wb_device *wb_dev = display;
  273. u16 hdisplay;
  274. int i;
  275. if (!drm_mode || !mode_info || !max_mixer_width || !display) {
  276. pr_err("invalid params\n");
  277. return -EINVAL;
  278. }
  279. hdisplay = drm_mode->hdisplay;
  280. /* find maximum display width to support */
  281. for (i = 0; i < wb_dev->count_modes; i++)
  282. hdisplay = max(hdisplay, wb_dev->modes[i].hdisplay);
  283. topology = &mode_info->topology;
  284. topology->num_lm = (max_mixer_width <= hdisplay) ? dual_lm : single_lm;
  285. topology->num_enc = no_enc;
  286. topology->num_intf = single_intf;
  287. mode_info->comp_info.comp_type = MSM_DISPLAY_COMPRESSION_NONE;
  288. mode_info->wide_bus_en = false;
  289. mode_info->comp_info.comp_ratio = MSM_DISPLAY_COMPRESSION_RATIO_NONE;
  290. return 0;
  291. }
  292. int sde_wb_connector_set_info_blob(struct drm_connector *connector,
  293. void *info, void *display, struct msm_mode_info *mode_info)
  294. {
  295. struct sde_wb_device *wb_dev = display;
  296. const struct sde_format_extended *format_list;
  297. if (!connector || !info || !display || !wb_dev->wb_cfg) {
  298. SDE_ERROR("invalid params\n");
  299. return -EINVAL;
  300. }
  301. format_list = wb_dev->wb_cfg->format_list;
  302. /*
  303. * Populate info buffer
  304. */
  305. if (format_list) {
  306. sde_kms_info_start(info, "pixel_formats");
  307. while (format_list->fourcc_format) {
  308. sde_kms_info_append_format(info,
  309. format_list->fourcc_format,
  310. format_list->modifier);
  311. ++format_list;
  312. }
  313. sde_kms_info_stop(info);
  314. }
  315. sde_kms_info_add_keyint(info,
  316. "wb_intf_index",
  317. wb_dev->wb_idx - WB_0);
  318. sde_kms_info_add_keyint(info,
  319. "maxlinewidth",
  320. wb_dev->wb_cfg->sblk->maxlinewidth);
  321. sde_kms_info_start(info, "features");
  322. if (wb_dev->wb_cfg && (wb_dev->wb_cfg->features & BIT(SDE_WB_UBWC)))
  323. sde_kms_info_append(info, "wb_ubwc");
  324. sde_kms_info_stop(info);
  325. return 0;
  326. }
  327. int sde_wb_connector_post_init(struct drm_connector *connector, void *display)
  328. {
  329. struct sde_connector *c_conn;
  330. struct sde_wb_device *wb_dev = display;
  331. static const struct drm_prop_enum_list e_fb_translation_mode[] = {
  332. {SDE_DRM_FB_NON_SEC, "non_sec"},
  333. {SDE_DRM_FB_SEC, "sec"},
  334. };
  335. if (!connector || !display || !wb_dev->wb_cfg) {
  336. SDE_ERROR("invalid params\n");
  337. return -EINVAL;
  338. }
  339. c_conn = to_sde_connector(connector);
  340. wb_dev->connector = connector;
  341. wb_dev->detect_status = connector_status_connected;
  342. /*
  343. * Add extra connector properties
  344. */
  345. msm_property_install_range(&c_conn->property_info, "FB_ID",
  346. 0x0, 0, ~0, 0, CONNECTOR_PROP_OUT_FB);
  347. msm_property_install_range(&c_conn->property_info, "DST_X",
  348. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_X);
  349. msm_property_install_range(&c_conn->property_info, "DST_Y",
  350. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_Y);
  351. msm_property_install_range(&c_conn->property_info, "DST_W",
  352. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_W);
  353. msm_property_install_range(&c_conn->property_info, "DST_H",
  354. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_H);
  355. msm_property_install_enum(&c_conn->property_info,
  356. "fb_translation_mode",
  357. 0x0,
  358. 0, e_fb_translation_mode,
  359. ARRAY_SIZE(e_fb_translation_mode),
  360. CONNECTOR_PROP_FB_TRANSLATION_MODE);
  361. return 0;
  362. }
  363. struct drm_framebuffer *sde_wb_get_output_fb(struct sde_wb_device *wb_dev)
  364. {
  365. struct drm_framebuffer *fb;
  366. if (!wb_dev || !wb_dev->connector) {
  367. SDE_ERROR("invalid params\n");
  368. return NULL;
  369. }
  370. SDE_DEBUG("\n");
  371. mutex_lock(&wb_dev->wb_lock);
  372. fb = sde_wb_connector_state_get_output_fb(wb_dev->connector->state);
  373. mutex_unlock(&wb_dev->wb_lock);
  374. return fb;
  375. }
  376. int sde_wb_get_output_roi(struct sde_wb_device *wb_dev, struct sde_rect *roi)
  377. {
  378. int rc;
  379. if (!wb_dev || !wb_dev->connector || !roi) {
  380. SDE_ERROR("invalid params\n");
  381. return -EINVAL;
  382. }
  383. SDE_DEBUG("\n");
  384. mutex_lock(&wb_dev->wb_lock);
  385. rc = sde_wb_connector_state_get_output_roi(
  386. wb_dev->connector->state, roi);
  387. mutex_unlock(&wb_dev->wb_lock);
  388. return rc;
  389. }
  390. u32 sde_wb_get_num_of_displays(void)
  391. {
  392. u32 count = 0;
  393. struct sde_wb_device *wb_dev;
  394. SDE_DEBUG("\n");
  395. mutex_lock(&sde_wb_list_lock);
  396. list_for_each_entry(wb_dev, &sde_wb_list, wb_list) {
  397. count++;
  398. }
  399. mutex_unlock(&sde_wb_list_lock);
  400. return count;
  401. }
  402. int wb_display_get_displays(void **display_array, u32 max_display_count)
  403. {
  404. struct sde_wb_device *curr;
  405. int i = 0;
  406. SDE_DEBUG("\n");
  407. if (!display_array || !max_display_count) {
  408. if (!display_array)
  409. SDE_ERROR("invalid param\n");
  410. return 0;
  411. }
  412. mutex_lock(&sde_wb_list_lock);
  413. list_for_each_entry(curr, &sde_wb_list, wb_list) {
  414. if (i >= max_display_count)
  415. break;
  416. display_array[i++] = curr;
  417. }
  418. mutex_unlock(&sde_wb_list_lock);
  419. return i;
  420. }
  421. int sde_wb_config(struct drm_device *drm_dev, void *data,
  422. struct drm_file *file_priv)
  423. {
  424. struct sde_drm_wb_cfg *config = data;
  425. struct msm_drm_private *priv;
  426. struct sde_wb_device *wb_dev = NULL;
  427. struct sde_wb_device *curr;
  428. struct drm_connector *connector;
  429. uint32_t flags;
  430. uint32_t connector_id;
  431. uint32_t count_modes;
  432. uint64_t modes;
  433. int rc;
  434. if (!drm_dev || !data) {
  435. SDE_ERROR("invalid params\n");
  436. return -EINVAL;
  437. }
  438. SDE_DEBUG("\n");
  439. flags = config->flags;
  440. connector_id = config->connector_id;
  441. count_modes = config->count_modes;
  442. modes = config->modes;
  443. priv = drm_dev->dev_private;
  444. connector = drm_connector_lookup(drm_dev, file_priv, connector_id);
  445. if (!connector) {
  446. SDE_ERROR("failed to find connector\n");
  447. rc = -ENOENT;
  448. goto fail;
  449. }
  450. mutex_lock(&sde_wb_list_lock);
  451. list_for_each_entry(curr, &sde_wb_list, wb_list) {
  452. if (curr->connector == connector) {
  453. wb_dev = curr;
  454. break;
  455. }
  456. }
  457. mutex_unlock(&sde_wb_list_lock);
  458. if (!wb_dev) {
  459. SDE_ERROR("failed to find wb device\n");
  460. rc = -ENOENT;
  461. goto fail;
  462. }
  463. mutex_lock(&wb_dev->wb_lock);
  464. rc = sde_wb_connector_set_modes(wb_dev, count_modes,
  465. (struct drm_mode_modeinfo __user *) (uintptr_t) modes,
  466. (flags & SDE_DRM_WB_CFG_FLAGS_CONNECTED) ? true : false);
  467. mutex_unlock(&wb_dev->wb_lock);
  468. drm_helper_hpd_irq_event(drm_dev);
  469. fail:
  470. return rc;
  471. }
  472. /**
  473. * _sde_wb_dev_init - perform device initialization
  474. * @wb_dev: Pointer to writeback device
  475. */
  476. static int _sde_wb_dev_init(struct sde_wb_device *wb_dev)
  477. {
  478. int rc = 0;
  479. if (!wb_dev) {
  480. SDE_ERROR("invalid params\n");
  481. return -EINVAL;
  482. }
  483. SDE_DEBUG("\n");
  484. return rc;
  485. }
  486. /**
  487. * _sde_wb_dev_deinit - perform device de-initialization
  488. * @wb_dev: Pointer to writeback device
  489. */
  490. static int _sde_wb_dev_deinit(struct sde_wb_device *wb_dev)
  491. {
  492. int rc = 0;
  493. if (!wb_dev) {
  494. SDE_ERROR("invalid params\n");
  495. return -EINVAL;
  496. }
  497. SDE_DEBUG("\n");
  498. return rc;
  499. }
  500. /**
  501. * sde_wb_bind - bind writeback device with controlling device
  502. * @dev: Pointer to base of platform device
  503. * @master: Pointer to container of drm device
  504. * @data: Pointer to private data
  505. * Returns: Zero on success
  506. */
  507. static int sde_wb_bind(struct device *dev, struct device *master, void *data)
  508. {
  509. struct sde_wb_device *wb_dev;
  510. if (!dev || !master) {
  511. SDE_ERROR("invalid params\n");
  512. return -EINVAL;
  513. }
  514. wb_dev = platform_get_drvdata(to_platform_device(dev));
  515. if (!wb_dev) {
  516. SDE_ERROR("invalid wb device\n");
  517. return -EINVAL;
  518. }
  519. SDE_DEBUG("\n");
  520. mutex_lock(&wb_dev->wb_lock);
  521. wb_dev->drm_dev = dev_get_drvdata(master);
  522. mutex_unlock(&wb_dev->wb_lock);
  523. return 0;
  524. }
  525. /**
  526. * sde_wb_unbind - unbind writeback from controlling device
  527. * @dev: Pointer to base of platform device
  528. * @master: Pointer to container of drm device
  529. * @data: Pointer to private data
  530. */
  531. static void sde_wb_unbind(struct device *dev,
  532. struct device *master, void *data)
  533. {
  534. struct sde_wb_device *wb_dev;
  535. if (!dev) {
  536. SDE_ERROR("invalid params\n");
  537. return;
  538. }
  539. wb_dev = platform_get_drvdata(to_platform_device(dev));
  540. if (!wb_dev) {
  541. SDE_ERROR("invalid wb device\n");
  542. return;
  543. }
  544. SDE_DEBUG("\n");
  545. mutex_lock(&wb_dev->wb_lock);
  546. wb_dev->drm_dev = NULL;
  547. mutex_unlock(&wb_dev->wb_lock);
  548. }
  549. static const struct component_ops sde_wb_comp_ops = {
  550. .bind = sde_wb_bind,
  551. .unbind = sde_wb_unbind,
  552. };
  553. /**
  554. * sde_wb_drm_init - perform DRM initialization
  555. * @wb_dev: Pointer to writeback device
  556. * @encoder: Pointer to associated encoder
  557. */
  558. int sde_wb_drm_init(struct sde_wb_device *wb_dev, struct drm_encoder *encoder)
  559. {
  560. int rc = 0;
  561. if (!wb_dev || !wb_dev->drm_dev || !encoder) {
  562. SDE_ERROR("invalid params\n");
  563. return -EINVAL;
  564. }
  565. SDE_DEBUG("\n");
  566. mutex_lock(&wb_dev->wb_lock);
  567. if (wb_dev->drm_dev->dev_private) {
  568. struct msm_drm_private *priv = wb_dev->drm_dev->dev_private;
  569. struct sde_kms *sde_kms = to_sde_kms(priv->kms);
  570. if (wb_dev->index < sde_kms->catalog->wb_count) {
  571. wb_dev->wb_idx = sde_kms->catalog->wb[wb_dev->index].id;
  572. wb_dev->wb_cfg = &sde_kms->catalog->wb[wb_dev->index];
  573. }
  574. }
  575. wb_dev->drm_dev = encoder->dev;
  576. wb_dev->encoder = encoder;
  577. mutex_unlock(&wb_dev->wb_lock);
  578. return rc;
  579. }
  580. int sde_wb_drm_deinit(struct sde_wb_device *wb_dev)
  581. {
  582. int rc = 0;
  583. if (!wb_dev) {
  584. SDE_ERROR("invalid params\n");
  585. return -EINVAL;
  586. }
  587. SDE_DEBUG("\n");
  588. return rc;
  589. }
  590. /**
  591. * sde_wb_probe - load writeback module
  592. * @pdev: Pointer to platform device
  593. */
  594. static int sde_wb_probe(struct platform_device *pdev)
  595. {
  596. struct sde_wb_device *wb_dev;
  597. int ret;
  598. wb_dev = devm_kzalloc(&pdev->dev, sizeof(*wb_dev), GFP_KERNEL);
  599. if (!wb_dev)
  600. return -ENOMEM;
  601. SDE_DEBUG("\n");
  602. ret = of_property_read_u32(pdev->dev.of_node, "cell-index",
  603. &wb_dev->index);
  604. if (ret) {
  605. SDE_DEBUG("cell index not set, default to 0\n");
  606. wb_dev->index = 0;
  607. }
  608. wb_dev->name = of_get_property(pdev->dev.of_node, "label", NULL);
  609. if (!wb_dev->name) {
  610. SDE_DEBUG("label not set, default to unknown\n");
  611. wb_dev->name = "unknown";
  612. }
  613. wb_dev->wb_idx = SDE_NONE;
  614. mutex_init(&wb_dev->wb_lock);
  615. platform_set_drvdata(pdev, wb_dev);
  616. mutex_lock(&sde_wb_list_lock);
  617. list_add(&wb_dev->wb_list, &sde_wb_list);
  618. mutex_unlock(&sde_wb_list_lock);
  619. if (!_sde_wb_dev_init(wb_dev)) {
  620. ret = component_add(&pdev->dev, &sde_wb_comp_ops);
  621. if (ret)
  622. pr_err("component add failed\n");
  623. }
  624. return ret;
  625. }
  626. /**
  627. * sde_wb_remove - unload writeback module
  628. * @pdev: Pointer to platform device
  629. */
  630. static int sde_wb_remove(struct platform_device *pdev)
  631. {
  632. struct sde_wb_device *wb_dev;
  633. struct sde_wb_device *curr, *next;
  634. wb_dev = platform_get_drvdata(pdev);
  635. if (!wb_dev)
  636. return 0;
  637. SDE_DEBUG("\n");
  638. (void)_sde_wb_dev_deinit(wb_dev);
  639. mutex_lock(&sde_wb_list_lock);
  640. list_for_each_entry_safe(curr, next, &sde_wb_list, wb_list) {
  641. if (curr == wb_dev) {
  642. list_del(&wb_dev->wb_list);
  643. break;
  644. }
  645. }
  646. mutex_unlock(&sde_wb_list_lock);
  647. kfree(wb_dev->modes);
  648. mutex_destroy(&wb_dev->wb_lock);
  649. platform_set_drvdata(pdev, NULL);
  650. devm_kfree(&pdev->dev, wb_dev);
  651. return 0;
  652. }
  653. static const struct of_device_id dt_match[] = {
  654. { .compatible = "qcom,wb-display"},
  655. {}
  656. };
  657. static struct platform_driver sde_wb_driver = {
  658. .probe = sde_wb_probe,
  659. .remove = sde_wb_remove,
  660. .driver = {
  661. .name = "sde_wb",
  662. .of_match_table = dt_match,
  663. .suppress_bind_attrs = true,
  664. },
  665. };
  666. static int __init sde_wb_register(void)
  667. {
  668. return platform_driver_register(&sde_wb_driver);
  669. }
  670. static void __exit sde_wb_unregister(void)
  671. {
  672. platform_driver_unregister(&sde_wb_driver);
  673. }
  674. module_init(sde_wb_register);
  675. module_exit(sde_wb_unregister);