sde_wb.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830
  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. modeinfo = kcalloc(count_modes,
  141. sizeof(struct drm_mode_modeinfo),
  142. GFP_KERNEL);
  143. if (!modeinfo) {
  144. SDE_ERROR("invalid params\n");
  145. ret = -ENOMEM;
  146. goto error;
  147. }
  148. if (copy_from_user(modeinfo, modes,
  149. count_modes *
  150. sizeof(struct drm_mode_modeinfo))) {
  151. SDE_ERROR("failed to copy modes\n");
  152. kfree(modeinfo);
  153. ret = -EFAULT;
  154. goto error;
  155. }
  156. for (i = 0; i < count_modes; i++) {
  157. struct drm_display_mode dispmode;
  158. memset(&dispmode, 0, sizeof(dispmode));
  159. ret = drm_mode_convert_umode(wb_dev->drm_dev,
  160. &dispmode, &modeinfo[i]);
  161. if (ret) {
  162. SDE_ERROR(
  163. "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",
  164. i,
  165. modeinfo[i].name,
  166. modeinfo[i].vrefresh,
  167. modeinfo[i].clock,
  168. modeinfo[i].hdisplay,
  169. modeinfo[i].hsync_start,
  170. modeinfo[i].hsync_end,
  171. modeinfo[i].htotal,
  172. modeinfo[i].vdisplay,
  173. modeinfo[i].vsync_start,
  174. modeinfo[i].vsync_end,
  175. modeinfo[i].vtotal,
  176. modeinfo[i].type,
  177. modeinfo[i].flags,
  178. dispmode.status,
  179. ret);
  180. kfree(modeinfo);
  181. goto error;
  182. }
  183. }
  184. }
  185. if (wb_dev->modes) {
  186. wb_dev->count_modes = 0;
  187. kfree(wb_dev->modes);
  188. wb_dev->modes = NULL;
  189. }
  190. wb_dev->count_modes = count_modes;
  191. wb_dev->modes = modeinfo;
  192. wb_dev->detect_status = connector_status_connected;
  193. } else {
  194. SDE_DEBUG("disconnect\n");
  195. if (wb_dev->modes) {
  196. wb_dev->count_modes = 0;
  197. kfree(wb_dev->modes);
  198. wb_dev->modes = NULL;
  199. }
  200. wb_dev->detect_status = connector_status_disconnected;
  201. }
  202. error:
  203. return ret;
  204. }
  205. int sde_wb_connector_set_property(struct drm_connector *connector,
  206. struct drm_connector_state *state,
  207. int property_index,
  208. uint64_t value,
  209. void *display)
  210. {
  211. struct sde_wb_device *wb_dev = display;
  212. struct drm_framebuffer *out_fb;
  213. int rc = 0;
  214. SDE_DEBUG("\n");
  215. if (state && (property_index == CONNECTOR_PROP_OUT_FB)) {
  216. const struct sde_format *sde_format;
  217. out_fb = sde_connector_get_out_fb(state);
  218. if (!out_fb)
  219. goto done;
  220. sde_format = sde_get_sde_format_ext(out_fb->format->format,
  221. out_fb->modifier);
  222. if (!sde_format) {
  223. SDE_ERROR("failed to get sde format\n");
  224. rc = -EINVAL;
  225. goto done;
  226. }
  227. if (!sde_wb_is_format_valid(wb_dev, out_fb->format->format,
  228. out_fb->modifier)) {
  229. SDE_ERROR("unsupported writeback format 0x%x/0x%llx\n",
  230. out_fb->format->format,
  231. out_fb->modifier);
  232. rc = -EINVAL;
  233. goto done;
  234. }
  235. }
  236. done:
  237. return rc;
  238. }
  239. int sde_wb_get_info(struct drm_connector *connector,
  240. struct msm_display_info *info, void *display)
  241. {
  242. struct sde_wb_device *wb_dev = display;
  243. if (!info || !wb_dev) {
  244. pr_err("invalid params\n");
  245. return -EINVAL;
  246. }
  247. memset(info, 0, sizeof(struct msm_display_info));
  248. info->intf_type = DRM_MODE_CONNECTOR_VIRTUAL;
  249. info->num_of_h_tiles = 1;
  250. info->h_tile_instance[0] = sde_wb_get_index(display);
  251. info->is_connected = true;
  252. info->capabilities = MSM_DISPLAY_CAP_HOT_PLUG | MSM_DISPLAY_CAP_EDID;
  253. info->max_width = (wb_dev->wb_cfg && wb_dev->wb_cfg->sblk) ?
  254. wb_dev->wb_cfg->sblk->maxlinewidth :
  255. SDE_WB_MODE_MAX_WIDTH;
  256. info->max_height = SDE_WB_MODE_MAX_HEIGHT;
  257. return 0;
  258. }
  259. int sde_wb_get_mode_info(struct drm_connector *connector,
  260. const struct drm_display_mode *drm_mode,
  261. struct msm_mode_info *mode_info,
  262. u32 max_mixer_width, void *display)
  263. {
  264. const u32 dual_lm = 2;
  265. const u32 single_lm = 1;
  266. const u32 single_intf = 1;
  267. const u32 no_enc = 0;
  268. struct msm_display_topology *topology;
  269. struct sde_wb_device *wb_dev = display;
  270. u16 hdisplay;
  271. int i;
  272. if (!drm_mode || !mode_info || !max_mixer_width || !display) {
  273. pr_err("invalid params\n");
  274. return -EINVAL;
  275. }
  276. hdisplay = drm_mode->hdisplay;
  277. /* find maximum display width to support */
  278. for (i = 0; i < wb_dev->count_modes; i++)
  279. hdisplay = max(hdisplay, wb_dev->modes[i].hdisplay);
  280. topology = &mode_info->topology;
  281. topology->num_lm = (max_mixer_width <= hdisplay) ? dual_lm : single_lm;
  282. topology->num_enc = no_enc;
  283. topology->num_intf = single_intf;
  284. mode_info->comp_info.comp_type = MSM_DISPLAY_COMPRESSION_NONE;
  285. mode_info->wide_bus_en = false;
  286. mode_info->comp_info.comp_ratio = MSM_DISPLAY_COMPRESSION_RATIO_NONE;
  287. return 0;
  288. }
  289. int sde_wb_connector_set_info_blob(struct drm_connector *connector,
  290. void *info, void *display, struct msm_mode_info *mode_info)
  291. {
  292. struct sde_wb_device *wb_dev = display;
  293. const struct sde_format_extended *format_list;
  294. if (!connector || !info || !display || !wb_dev->wb_cfg) {
  295. SDE_ERROR("invalid params\n");
  296. return -EINVAL;
  297. }
  298. format_list = wb_dev->wb_cfg->format_list;
  299. /*
  300. * Populate info buffer
  301. */
  302. if (format_list) {
  303. sde_kms_info_start(info, "pixel_formats");
  304. while (format_list->fourcc_format) {
  305. sde_kms_info_append_format(info,
  306. format_list->fourcc_format,
  307. format_list->modifier);
  308. ++format_list;
  309. }
  310. sde_kms_info_stop(info);
  311. }
  312. sde_kms_info_add_keyint(info,
  313. "wb_intf_index",
  314. wb_dev->wb_idx - WB_0);
  315. sde_kms_info_add_keyint(info,
  316. "maxlinewidth",
  317. wb_dev->wb_cfg->sblk->maxlinewidth);
  318. sde_kms_info_start(info, "features");
  319. if (wb_dev->wb_cfg && (wb_dev->wb_cfg->features & BIT(SDE_WB_UBWC)))
  320. sde_kms_info_append(info, "wb_ubwc");
  321. sde_kms_info_stop(info);
  322. return 0;
  323. }
  324. int sde_wb_connector_post_init(struct drm_connector *connector, void *display)
  325. {
  326. struct sde_connector *c_conn;
  327. struct sde_wb_device *wb_dev = display;
  328. static const struct drm_prop_enum_list e_fb_translation_mode[] = {
  329. {SDE_DRM_FB_NON_SEC, "non_sec"},
  330. {SDE_DRM_FB_SEC, "sec"},
  331. };
  332. if (!connector || !display || !wb_dev->wb_cfg) {
  333. SDE_ERROR("invalid params\n");
  334. return -EINVAL;
  335. }
  336. c_conn = to_sde_connector(connector);
  337. wb_dev->connector = connector;
  338. wb_dev->detect_status = connector_status_connected;
  339. /*
  340. * Add extra connector properties
  341. */
  342. msm_property_install_range(&c_conn->property_info, "FB_ID",
  343. 0x0, 0, ~0, 0, CONNECTOR_PROP_OUT_FB);
  344. msm_property_install_range(&c_conn->property_info, "DST_X",
  345. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_X);
  346. msm_property_install_range(&c_conn->property_info, "DST_Y",
  347. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_Y);
  348. msm_property_install_range(&c_conn->property_info, "DST_W",
  349. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_W);
  350. msm_property_install_range(&c_conn->property_info, "DST_H",
  351. 0x0, 0, UINT_MAX, 0, CONNECTOR_PROP_DST_H);
  352. msm_property_install_enum(&c_conn->property_info,
  353. "fb_translation_mode",
  354. 0x0,
  355. 0, e_fb_translation_mode,
  356. ARRAY_SIZE(e_fb_translation_mode),
  357. CONNECTOR_PROP_FB_TRANSLATION_MODE);
  358. return 0;
  359. }
  360. struct drm_framebuffer *sde_wb_get_output_fb(struct sde_wb_device *wb_dev)
  361. {
  362. struct drm_framebuffer *fb;
  363. if (!wb_dev || !wb_dev->connector) {
  364. SDE_ERROR("invalid params\n");
  365. return NULL;
  366. }
  367. SDE_DEBUG("\n");
  368. mutex_lock(&wb_dev->wb_lock);
  369. fb = sde_wb_connector_state_get_output_fb(wb_dev->connector->state);
  370. mutex_unlock(&wb_dev->wb_lock);
  371. return fb;
  372. }
  373. int sde_wb_get_output_roi(struct sde_wb_device *wb_dev, struct sde_rect *roi)
  374. {
  375. int rc;
  376. if (!wb_dev || !wb_dev->connector || !roi) {
  377. SDE_ERROR("invalid params\n");
  378. return -EINVAL;
  379. }
  380. SDE_DEBUG("\n");
  381. mutex_lock(&wb_dev->wb_lock);
  382. rc = sde_wb_connector_state_get_output_roi(
  383. wb_dev->connector->state, roi);
  384. mutex_unlock(&wb_dev->wb_lock);
  385. return rc;
  386. }
  387. u32 sde_wb_get_num_of_displays(void)
  388. {
  389. u32 count = 0;
  390. struct sde_wb_device *wb_dev;
  391. SDE_DEBUG("\n");
  392. mutex_lock(&sde_wb_list_lock);
  393. list_for_each_entry(wb_dev, &sde_wb_list, wb_list) {
  394. count++;
  395. }
  396. mutex_unlock(&sde_wb_list_lock);
  397. return count;
  398. }
  399. int wb_display_get_displays(void **display_array, u32 max_display_count)
  400. {
  401. struct sde_wb_device *curr;
  402. int i = 0;
  403. SDE_DEBUG("\n");
  404. if (!display_array || !max_display_count) {
  405. if (!display_array)
  406. SDE_ERROR("invalid param\n");
  407. return 0;
  408. }
  409. mutex_lock(&sde_wb_list_lock);
  410. list_for_each_entry(curr, &sde_wb_list, wb_list) {
  411. if (i >= max_display_count)
  412. break;
  413. display_array[i++] = curr;
  414. }
  415. mutex_unlock(&sde_wb_list_lock);
  416. return i;
  417. }
  418. int sde_wb_config(struct drm_device *drm_dev, void *data,
  419. struct drm_file *file_priv)
  420. {
  421. struct sde_drm_wb_cfg *config = data;
  422. struct msm_drm_private *priv;
  423. struct sde_wb_device *wb_dev = NULL;
  424. struct sde_wb_device *curr;
  425. struct drm_connector *connector;
  426. uint32_t flags;
  427. uint32_t connector_id;
  428. uint32_t count_modes;
  429. uint64_t modes;
  430. int rc;
  431. if (!drm_dev || !data) {
  432. SDE_ERROR("invalid params\n");
  433. return -EINVAL;
  434. }
  435. SDE_DEBUG("\n");
  436. flags = config->flags;
  437. connector_id = config->connector_id;
  438. count_modes = config->count_modes;
  439. modes = config->modes;
  440. priv = drm_dev->dev_private;
  441. connector = drm_connector_lookup(drm_dev, file_priv, connector_id);
  442. if (!connector) {
  443. SDE_ERROR("failed to find connector\n");
  444. rc = -ENOENT;
  445. goto fail;
  446. }
  447. mutex_lock(&sde_wb_list_lock);
  448. list_for_each_entry(curr, &sde_wb_list, wb_list) {
  449. if (curr->connector == connector) {
  450. wb_dev = curr;
  451. break;
  452. }
  453. }
  454. mutex_unlock(&sde_wb_list_lock);
  455. if (!wb_dev) {
  456. SDE_ERROR("failed to find wb device\n");
  457. rc = -ENOENT;
  458. goto fail;
  459. }
  460. mutex_lock(&wb_dev->wb_lock);
  461. rc = sde_wb_connector_set_modes(wb_dev, count_modes,
  462. (struct drm_mode_modeinfo __user *) (uintptr_t) modes,
  463. (flags & SDE_DRM_WB_CFG_FLAGS_CONNECTED) ? true : false);
  464. mutex_unlock(&wb_dev->wb_lock);
  465. drm_helper_hpd_irq_event(drm_dev);
  466. fail:
  467. return rc;
  468. }
  469. /**
  470. * _sde_wb_dev_init - perform device initialization
  471. * @wb_dev: Pointer to writeback device
  472. */
  473. static int _sde_wb_dev_init(struct sde_wb_device *wb_dev)
  474. {
  475. int rc = 0;
  476. if (!wb_dev) {
  477. SDE_ERROR("invalid params\n");
  478. return -EINVAL;
  479. }
  480. SDE_DEBUG("\n");
  481. return rc;
  482. }
  483. /**
  484. * _sde_wb_dev_deinit - perform device de-initialization
  485. * @wb_dev: Pointer to writeback device
  486. */
  487. static int _sde_wb_dev_deinit(struct sde_wb_device *wb_dev)
  488. {
  489. int rc = 0;
  490. if (!wb_dev) {
  491. SDE_ERROR("invalid params\n");
  492. return -EINVAL;
  493. }
  494. SDE_DEBUG("\n");
  495. return rc;
  496. }
  497. /**
  498. * sde_wb_bind - bind writeback device with controlling device
  499. * @dev: Pointer to base of platform device
  500. * @master: Pointer to container of drm device
  501. * @data: Pointer to private data
  502. * Returns: Zero on success
  503. */
  504. static int sde_wb_bind(struct device *dev, struct device *master, void *data)
  505. {
  506. struct sde_wb_device *wb_dev;
  507. if (!dev || !master) {
  508. SDE_ERROR("invalid params\n");
  509. return -EINVAL;
  510. }
  511. wb_dev = platform_get_drvdata(to_platform_device(dev));
  512. if (!wb_dev) {
  513. SDE_ERROR("invalid wb device\n");
  514. return -EINVAL;
  515. }
  516. SDE_DEBUG("\n");
  517. mutex_lock(&wb_dev->wb_lock);
  518. wb_dev->drm_dev = dev_get_drvdata(master);
  519. mutex_unlock(&wb_dev->wb_lock);
  520. return 0;
  521. }
  522. /**
  523. * sde_wb_unbind - unbind writeback from controlling device
  524. * @dev: Pointer to base of platform device
  525. * @master: Pointer to container of drm device
  526. * @data: Pointer to private data
  527. */
  528. static void sde_wb_unbind(struct device *dev,
  529. struct device *master, void *data)
  530. {
  531. struct sde_wb_device *wb_dev;
  532. if (!dev) {
  533. SDE_ERROR("invalid params\n");
  534. return;
  535. }
  536. wb_dev = platform_get_drvdata(to_platform_device(dev));
  537. if (!wb_dev) {
  538. SDE_ERROR("invalid wb device\n");
  539. return;
  540. }
  541. SDE_DEBUG("\n");
  542. mutex_lock(&wb_dev->wb_lock);
  543. wb_dev->drm_dev = NULL;
  544. mutex_unlock(&wb_dev->wb_lock);
  545. }
  546. static const struct component_ops sde_wb_comp_ops = {
  547. .bind = sde_wb_bind,
  548. .unbind = sde_wb_unbind,
  549. };
  550. /**
  551. * sde_wb_drm_init - perform DRM initialization
  552. * @wb_dev: Pointer to writeback device
  553. * @encoder: Pointer to associated encoder
  554. */
  555. int sde_wb_drm_init(struct sde_wb_device *wb_dev, struct drm_encoder *encoder)
  556. {
  557. int rc = 0;
  558. if (!wb_dev || !wb_dev->drm_dev || !encoder) {
  559. SDE_ERROR("invalid params\n");
  560. return -EINVAL;
  561. }
  562. SDE_DEBUG("\n");
  563. mutex_lock(&wb_dev->wb_lock);
  564. if (wb_dev->drm_dev->dev_private) {
  565. struct msm_drm_private *priv = wb_dev->drm_dev->dev_private;
  566. struct sde_kms *sde_kms = to_sde_kms(priv->kms);
  567. if (wb_dev->index < sde_kms->catalog->wb_count) {
  568. wb_dev->wb_idx = sde_kms->catalog->wb[wb_dev->index].id;
  569. wb_dev->wb_cfg = &sde_kms->catalog->wb[wb_dev->index];
  570. }
  571. }
  572. wb_dev->drm_dev = encoder->dev;
  573. wb_dev->encoder = encoder;
  574. mutex_unlock(&wb_dev->wb_lock);
  575. return rc;
  576. }
  577. int sde_wb_drm_deinit(struct sde_wb_device *wb_dev)
  578. {
  579. int rc = 0;
  580. if (!wb_dev) {
  581. SDE_ERROR("invalid params\n");
  582. return -EINVAL;
  583. }
  584. SDE_DEBUG("\n");
  585. return rc;
  586. }
  587. /**
  588. * sde_wb_probe - load writeback module
  589. * @pdev: Pointer to platform device
  590. */
  591. static int sde_wb_probe(struct platform_device *pdev)
  592. {
  593. struct sde_wb_device *wb_dev;
  594. int ret;
  595. wb_dev = devm_kzalloc(&pdev->dev, sizeof(*wb_dev), GFP_KERNEL);
  596. if (!wb_dev)
  597. return -ENOMEM;
  598. SDE_DEBUG("\n");
  599. ret = of_property_read_u32(pdev->dev.of_node, "cell-index",
  600. &wb_dev->index);
  601. if (ret) {
  602. SDE_DEBUG("cell index not set, default to 0\n");
  603. wb_dev->index = 0;
  604. }
  605. wb_dev->name = of_get_property(pdev->dev.of_node, "label", NULL);
  606. if (!wb_dev->name) {
  607. SDE_DEBUG("label not set, default to unknown\n");
  608. wb_dev->name = "unknown";
  609. }
  610. wb_dev->wb_idx = SDE_NONE;
  611. mutex_init(&wb_dev->wb_lock);
  612. platform_set_drvdata(pdev, wb_dev);
  613. mutex_lock(&sde_wb_list_lock);
  614. list_add(&wb_dev->wb_list, &sde_wb_list);
  615. mutex_unlock(&sde_wb_list_lock);
  616. if (!_sde_wb_dev_init(wb_dev)) {
  617. ret = component_add(&pdev->dev, &sde_wb_comp_ops);
  618. if (ret)
  619. pr_err("component add failed\n");
  620. }
  621. return ret;
  622. }
  623. /**
  624. * sde_wb_remove - unload writeback module
  625. * @pdev: Pointer to platform device
  626. */
  627. static int sde_wb_remove(struct platform_device *pdev)
  628. {
  629. struct sde_wb_device *wb_dev;
  630. struct sde_wb_device *curr, *next;
  631. wb_dev = platform_get_drvdata(pdev);
  632. if (!wb_dev)
  633. return 0;
  634. SDE_DEBUG("\n");
  635. (void)_sde_wb_dev_deinit(wb_dev);
  636. mutex_lock(&sde_wb_list_lock);
  637. list_for_each_entry_safe(curr, next, &sde_wb_list, wb_list) {
  638. if (curr == wb_dev) {
  639. list_del(&wb_dev->wb_list);
  640. break;
  641. }
  642. }
  643. mutex_unlock(&sde_wb_list_lock);
  644. kfree(wb_dev->modes);
  645. mutex_destroy(&wb_dev->wb_lock);
  646. platform_set_drvdata(pdev, NULL);
  647. devm_kfree(&pdev->dev, wb_dev);
  648. return 0;
  649. }
  650. static const struct of_device_id dt_match[] = {
  651. { .compatible = "qcom,wb-display"},
  652. {}
  653. };
  654. static struct platform_driver sde_wb_driver = {
  655. .probe = sde_wb_probe,
  656. .remove = sde_wb_remove,
  657. .driver = {
  658. .name = "sde_wb",
  659. .of_match_table = dt_match,
  660. .suppress_bind_attrs = true,
  661. },
  662. };
  663. static int __init sde_wb_register(void)
  664. {
  665. return platform_driver_register(&sde_wb_driver);
  666. }
  667. static void __exit sde_wb_unregister(void)
  668. {
  669. platform_driver_unregister(&sde_wb_driver);
  670. }
  671. module_init(sde_wb_register);
  672. module_exit(sde_wb_unregister);