mtk_drm_drv.c 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704
  1. /*
  2. * Copyright (c) 2015 MediaTek Inc.
  3. * Author: YT SHEN <yt.shen@mediatek.com>
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License version 2 as
  7. * published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. */
  14. #include <drm/drmP.h>
  15. #include <drm/drm_atomic.h>
  16. #include <drm/drm_atomic_helper.h>
  17. #include <drm/drm_crtc_helper.h>
  18. #include <drm/drm_gem.h>
  19. #include <drm/drm_gem_cma_helper.h>
  20. #include <drm/drm_of.h>
  21. #include <linux/component.h>
  22. #include <linux/iommu.h>
  23. #include <linux/of_address.h>
  24. #include <linux/of_platform.h>
  25. #include <linux/pm_runtime.h>
  26. #include "mtk_drm_crtc.h"
  27. #include "mtk_drm_ddp.h"
  28. #include "mtk_drm_ddp_comp.h"
  29. #include "mtk_drm_drv.h"
  30. #include "mtk_drm_fb.h"
  31. #include "mtk_drm_gem.h"
  32. #define DRIVER_NAME "mediatek"
  33. #define DRIVER_DESC "Mediatek SoC DRM"
  34. #define DRIVER_DATE "20150513"
  35. #define DRIVER_MAJOR 1
  36. #define DRIVER_MINOR 0
  37. static void mtk_atomic_schedule(struct mtk_drm_private *private,
  38. struct drm_atomic_state *state)
  39. {
  40. private->commit.state = state;
  41. schedule_work(&private->commit.work);
  42. }
  43. static void mtk_atomic_wait_for_fences(struct drm_atomic_state *state)
  44. {
  45. struct drm_plane *plane;
  46. struct drm_plane_state *new_plane_state;
  47. int i;
  48. for_each_new_plane_in_state(state, plane, new_plane_state, i)
  49. mtk_fb_wait(new_plane_state->fb);
  50. }
  51. static void mtk_atomic_complete(struct mtk_drm_private *private,
  52. struct drm_atomic_state *state)
  53. {
  54. struct drm_device *drm = private->drm;
  55. mtk_atomic_wait_for_fences(state);
  56. /*
  57. * Mediatek drm supports runtime PM, so plane registers cannot be
  58. * written when their crtc is disabled.
  59. *
  60. * The comment for drm_atomic_helper_commit states:
  61. * For drivers supporting runtime PM the recommended sequence is
  62. *
  63. * drm_atomic_helper_commit_modeset_disables(dev, state);
  64. * drm_atomic_helper_commit_modeset_enables(dev, state);
  65. * drm_atomic_helper_commit_planes(dev, state,
  66. * DRM_PLANE_COMMIT_ACTIVE_ONLY);
  67. *
  68. * See the kerneldoc entries for these three functions for more details.
  69. */
  70. drm_atomic_helper_commit_modeset_disables(drm, state);
  71. drm_atomic_helper_commit_modeset_enables(drm, state);
  72. drm_atomic_helper_commit_planes(drm, state,
  73. DRM_PLANE_COMMIT_ACTIVE_ONLY);
  74. drm_atomic_helper_wait_for_vblanks(drm, state);
  75. drm_atomic_helper_cleanup_planes(drm, state);
  76. drm_atomic_state_put(state);
  77. }
  78. static void mtk_atomic_work(struct work_struct *work)
  79. {
  80. struct mtk_drm_private *private = container_of(work,
  81. struct mtk_drm_private, commit.work);
  82. mtk_atomic_complete(private, private->commit.state);
  83. }
  84. static int mtk_atomic_commit(struct drm_device *drm,
  85. struct drm_atomic_state *state,
  86. bool async)
  87. {
  88. struct mtk_drm_private *private = drm->dev_private;
  89. int ret;
  90. ret = drm_atomic_helper_prepare_planes(drm, state);
  91. if (ret)
  92. return ret;
  93. mutex_lock(&private->commit.lock);
  94. flush_work(&private->commit.work);
  95. ret = drm_atomic_helper_swap_state(state, true);
  96. if (ret) {
  97. mutex_unlock(&private->commit.lock);
  98. drm_atomic_helper_cleanup_planes(drm, state);
  99. return ret;
  100. }
  101. drm_atomic_state_get(state);
  102. if (async)
  103. mtk_atomic_schedule(private, state);
  104. else
  105. mtk_atomic_complete(private, state);
  106. mutex_unlock(&private->commit.lock);
  107. return 0;
  108. }
  109. static const struct drm_mode_config_funcs mtk_drm_mode_config_funcs = {
  110. .fb_create = mtk_drm_mode_fb_create,
  111. .atomic_check = drm_atomic_helper_check,
  112. .atomic_commit = mtk_atomic_commit,
  113. };
  114. static const enum mtk_ddp_comp_id mt2701_mtk_ddp_main[] = {
  115. DDP_COMPONENT_OVL0,
  116. DDP_COMPONENT_RDMA0,
  117. DDP_COMPONENT_COLOR0,
  118. DDP_COMPONENT_BLS,
  119. DDP_COMPONENT_DSI0,
  120. };
  121. static const enum mtk_ddp_comp_id mt2701_mtk_ddp_ext[] = {
  122. DDP_COMPONENT_RDMA1,
  123. DDP_COMPONENT_DPI0,
  124. };
  125. static const enum mtk_ddp_comp_id mt2712_mtk_ddp_main[] = {
  126. DDP_COMPONENT_OVL0,
  127. DDP_COMPONENT_COLOR0,
  128. DDP_COMPONENT_AAL0,
  129. DDP_COMPONENT_OD0,
  130. DDP_COMPONENT_RDMA0,
  131. DDP_COMPONENT_DPI0,
  132. DDP_COMPONENT_PWM0,
  133. };
  134. static const enum mtk_ddp_comp_id mt2712_mtk_ddp_ext[] = {
  135. DDP_COMPONENT_OVL1,
  136. DDP_COMPONENT_COLOR1,
  137. DDP_COMPONENT_AAL1,
  138. DDP_COMPONENT_OD1,
  139. DDP_COMPONENT_RDMA1,
  140. DDP_COMPONENT_DPI1,
  141. DDP_COMPONENT_PWM1,
  142. };
  143. static const enum mtk_ddp_comp_id mt2712_mtk_ddp_third[] = {
  144. DDP_COMPONENT_RDMA2,
  145. DDP_COMPONENT_DSI3,
  146. DDP_COMPONENT_PWM2,
  147. };
  148. static const enum mtk_ddp_comp_id mt8173_mtk_ddp_main[] = {
  149. DDP_COMPONENT_OVL0,
  150. DDP_COMPONENT_COLOR0,
  151. DDP_COMPONENT_AAL0,
  152. DDP_COMPONENT_OD0,
  153. DDP_COMPONENT_RDMA0,
  154. DDP_COMPONENT_UFOE,
  155. DDP_COMPONENT_DSI0,
  156. DDP_COMPONENT_PWM0,
  157. };
  158. static const enum mtk_ddp_comp_id mt8173_mtk_ddp_ext[] = {
  159. DDP_COMPONENT_OVL1,
  160. DDP_COMPONENT_COLOR1,
  161. DDP_COMPONENT_GAMMA,
  162. DDP_COMPONENT_RDMA1,
  163. DDP_COMPONENT_DPI0,
  164. };
  165. static const struct mtk_mmsys_driver_data mt2701_mmsys_driver_data = {
  166. .main_path = mt2701_mtk_ddp_main,
  167. .main_len = ARRAY_SIZE(mt2701_mtk_ddp_main),
  168. .ext_path = mt2701_mtk_ddp_ext,
  169. .ext_len = ARRAY_SIZE(mt2701_mtk_ddp_ext),
  170. .shadow_register = true,
  171. };
  172. static const struct mtk_mmsys_driver_data mt2712_mmsys_driver_data = {
  173. .main_path = mt2712_mtk_ddp_main,
  174. .main_len = ARRAY_SIZE(mt2712_mtk_ddp_main),
  175. .ext_path = mt2712_mtk_ddp_ext,
  176. .ext_len = ARRAY_SIZE(mt2712_mtk_ddp_ext),
  177. .third_path = mt2712_mtk_ddp_third,
  178. .third_len = ARRAY_SIZE(mt2712_mtk_ddp_third),
  179. };
  180. static const struct mtk_mmsys_driver_data mt8173_mmsys_driver_data = {
  181. .main_path = mt8173_mtk_ddp_main,
  182. .main_len = ARRAY_SIZE(mt8173_mtk_ddp_main),
  183. .ext_path = mt8173_mtk_ddp_ext,
  184. .ext_len = ARRAY_SIZE(mt8173_mtk_ddp_ext),
  185. };
  186. static int mtk_drm_kms_init(struct drm_device *drm)
  187. {
  188. struct mtk_drm_private *private = drm->dev_private;
  189. struct platform_device *pdev;
  190. struct device_node *np;
  191. struct device *dma_dev;
  192. int ret;
  193. if (!iommu_present(&platform_bus_type))
  194. return -EPROBE_DEFER;
  195. pdev = of_find_device_by_node(private->mutex_node);
  196. if (!pdev) {
  197. dev_err(drm->dev, "Waiting for disp-mutex device %pOF\n",
  198. private->mutex_node);
  199. of_node_put(private->mutex_node);
  200. return -EPROBE_DEFER;
  201. }
  202. private->mutex_dev = &pdev->dev;
  203. drm_mode_config_init(drm);
  204. drm->mode_config.min_width = 64;
  205. drm->mode_config.min_height = 64;
  206. /*
  207. * set max width and height as default value(4096x4096).
  208. * this value would be used to check framebuffer size limitation
  209. * at drm_mode_addfb().
  210. */
  211. drm->mode_config.max_width = 4096;
  212. drm->mode_config.max_height = 4096;
  213. drm->mode_config.funcs = &mtk_drm_mode_config_funcs;
  214. ret = component_bind_all(drm->dev, drm);
  215. if (ret)
  216. goto err_config_cleanup;
  217. /*
  218. * We currently support two fixed data streams, each optional,
  219. * and each statically assigned to a crtc:
  220. * OVL0 -> COLOR0 -> AAL -> OD -> RDMA0 -> UFOE -> DSI0 ...
  221. */
  222. ret = mtk_drm_crtc_create(drm, private->data->main_path,
  223. private->data->main_len);
  224. if (ret < 0)
  225. goto err_component_unbind;
  226. /* ... and OVL1 -> COLOR1 -> GAMMA -> RDMA1 -> DPI0. */
  227. ret = mtk_drm_crtc_create(drm, private->data->ext_path,
  228. private->data->ext_len);
  229. if (ret < 0)
  230. goto err_component_unbind;
  231. ret = mtk_drm_crtc_create(drm, private->data->third_path,
  232. private->data->third_len);
  233. if (ret < 0)
  234. goto err_component_unbind;
  235. /* Use OVL device for all DMA memory allocations */
  236. np = private->comp_node[private->data->main_path[0]] ?:
  237. private->comp_node[private->data->ext_path[0]];
  238. pdev = of_find_device_by_node(np);
  239. if (!pdev) {
  240. ret = -ENODEV;
  241. dev_err(drm->dev, "Need at least one OVL device\n");
  242. goto err_component_unbind;
  243. }
  244. dma_dev = &pdev->dev;
  245. private->dma_dev = dma_dev;
  246. /*
  247. * Configure the DMA segment size to make sure we get contiguous IOVA
  248. * when importing PRIME buffers.
  249. */
  250. if (!dma_dev->dma_parms) {
  251. private->dma_parms_allocated = true;
  252. dma_dev->dma_parms =
  253. devm_kzalloc(drm->dev, sizeof(*dma_dev->dma_parms),
  254. GFP_KERNEL);
  255. }
  256. if (!dma_dev->dma_parms) {
  257. ret = -ENOMEM;
  258. goto err_component_unbind;
  259. }
  260. ret = dma_set_max_seg_size(dma_dev, (unsigned int)DMA_BIT_MASK(32));
  261. if (ret) {
  262. dev_err(dma_dev, "Failed to set DMA segment size\n");
  263. goto err_unset_dma_parms;
  264. }
  265. /*
  266. * We don't use the drm_irq_install() helpers provided by the DRM
  267. * core, so we need to set this manually in order to allow the
  268. * DRM_IOCTL_WAIT_VBLANK to operate correctly.
  269. */
  270. drm->irq_enabled = true;
  271. ret = drm_vblank_init(drm, MAX_CRTC);
  272. if (ret < 0)
  273. goto err_unset_dma_parms;
  274. drm_kms_helper_poll_init(drm);
  275. drm_mode_config_reset(drm);
  276. return 0;
  277. err_unset_dma_parms:
  278. if (private->dma_parms_allocated)
  279. dma_dev->dma_parms = NULL;
  280. err_component_unbind:
  281. component_unbind_all(drm->dev, drm);
  282. err_config_cleanup:
  283. drm_mode_config_cleanup(drm);
  284. return ret;
  285. }
  286. static void mtk_drm_kms_deinit(struct drm_device *drm)
  287. {
  288. struct mtk_drm_private *private = drm->dev_private;
  289. drm_kms_helper_poll_fini(drm);
  290. drm_atomic_helper_shutdown(drm);
  291. if (private->dma_parms_allocated)
  292. private->dma_dev->dma_parms = NULL;
  293. component_unbind_all(drm->dev, drm);
  294. drm_mode_config_cleanup(drm);
  295. }
  296. static const struct file_operations mtk_drm_fops = {
  297. .owner = THIS_MODULE,
  298. .open = drm_open,
  299. .release = drm_release,
  300. .unlocked_ioctl = drm_ioctl,
  301. .mmap = mtk_drm_gem_mmap,
  302. .poll = drm_poll,
  303. .read = drm_read,
  304. .compat_ioctl = drm_compat_ioctl,
  305. };
  306. /*
  307. * We need to override this because the device used to import the memory is
  308. * not dev->dev, as drm_gem_prime_import() expects.
  309. */
  310. struct drm_gem_object *mtk_drm_gem_prime_import(struct drm_device *dev,
  311. struct dma_buf *dma_buf)
  312. {
  313. struct mtk_drm_private *private = dev->dev_private;
  314. return drm_gem_prime_import_dev(dev, dma_buf, private->dma_dev);
  315. }
  316. static struct drm_driver mtk_drm_driver = {
  317. .driver_features = DRIVER_MODESET | DRIVER_GEM | DRIVER_PRIME |
  318. DRIVER_ATOMIC,
  319. .gem_free_object_unlocked = mtk_drm_gem_free_object,
  320. .gem_vm_ops = &drm_gem_cma_vm_ops,
  321. .dumb_create = mtk_drm_gem_dumb_create,
  322. .prime_handle_to_fd = drm_gem_prime_handle_to_fd,
  323. .prime_fd_to_handle = drm_gem_prime_fd_to_handle,
  324. .gem_prime_export = drm_gem_prime_export,
  325. .gem_prime_import = mtk_drm_gem_prime_import,
  326. .gem_prime_get_sg_table = mtk_gem_prime_get_sg_table,
  327. .gem_prime_import_sg_table = mtk_gem_prime_import_sg_table,
  328. .gem_prime_mmap = mtk_drm_gem_mmap_buf,
  329. .fops = &mtk_drm_fops,
  330. .name = DRIVER_NAME,
  331. .desc = DRIVER_DESC,
  332. .date = DRIVER_DATE,
  333. .major = DRIVER_MAJOR,
  334. .minor = DRIVER_MINOR,
  335. };
  336. static int compare_of(struct device *dev, void *data)
  337. {
  338. return dev->of_node == data;
  339. }
  340. static int mtk_drm_bind(struct device *dev)
  341. {
  342. struct mtk_drm_private *private = dev_get_drvdata(dev);
  343. struct drm_device *drm;
  344. int ret;
  345. drm = drm_dev_alloc(&mtk_drm_driver, dev);
  346. if (IS_ERR(drm))
  347. return PTR_ERR(drm);
  348. drm->dev_private = private;
  349. private->drm = drm;
  350. ret = mtk_drm_kms_init(drm);
  351. if (ret < 0)
  352. goto err_free;
  353. ret = drm_dev_register(drm, 0);
  354. if (ret < 0)
  355. goto err_deinit;
  356. return 0;
  357. err_deinit:
  358. mtk_drm_kms_deinit(drm);
  359. err_free:
  360. drm_dev_put(drm);
  361. return ret;
  362. }
  363. static void mtk_drm_unbind(struct device *dev)
  364. {
  365. struct mtk_drm_private *private = dev_get_drvdata(dev);
  366. drm_dev_unregister(private->drm);
  367. mtk_drm_kms_deinit(private->drm);
  368. drm_dev_put(private->drm);
  369. private->num_pipes = 0;
  370. private->drm = NULL;
  371. }
  372. static const struct component_master_ops mtk_drm_ops = {
  373. .bind = mtk_drm_bind,
  374. .unbind = mtk_drm_unbind,
  375. };
  376. static const struct of_device_id mtk_ddp_comp_dt_ids[] = {
  377. { .compatible = "mediatek,mt2701-disp-ovl",
  378. .data = (void *)MTK_DISP_OVL },
  379. { .compatible = "mediatek,mt8173-disp-ovl",
  380. .data = (void *)MTK_DISP_OVL },
  381. { .compatible = "mediatek,mt2701-disp-rdma",
  382. .data = (void *)MTK_DISP_RDMA },
  383. { .compatible = "mediatek,mt8173-disp-rdma",
  384. .data = (void *)MTK_DISP_RDMA },
  385. { .compatible = "mediatek,mt8173-disp-wdma",
  386. .data = (void *)MTK_DISP_WDMA },
  387. { .compatible = "mediatek,mt2701-disp-color",
  388. .data = (void *)MTK_DISP_COLOR },
  389. { .compatible = "mediatek,mt8173-disp-color",
  390. .data = (void *)MTK_DISP_COLOR },
  391. { .compatible = "mediatek,mt8173-disp-aal",
  392. .data = (void *)MTK_DISP_AAL},
  393. { .compatible = "mediatek,mt8173-disp-gamma",
  394. .data = (void *)MTK_DISP_GAMMA, },
  395. { .compatible = "mediatek,mt8173-disp-ufoe",
  396. .data = (void *)MTK_DISP_UFOE },
  397. { .compatible = "mediatek,mt2701-dsi",
  398. .data = (void *)MTK_DSI },
  399. { .compatible = "mediatek,mt8173-dsi",
  400. .data = (void *)MTK_DSI },
  401. { .compatible = "mediatek,mt8173-dpi",
  402. .data = (void *)MTK_DPI },
  403. { .compatible = "mediatek,mt2701-disp-mutex",
  404. .data = (void *)MTK_DISP_MUTEX },
  405. { .compatible = "mediatek,mt2712-disp-mutex",
  406. .data = (void *)MTK_DISP_MUTEX },
  407. { .compatible = "mediatek,mt8173-disp-mutex",
  408. .data = (void *)MTK_DISP_MUTEX },
  409. { .compatible = "mediatek,mt2701-disp-pwm",
  410. .data = (void *)MTK_DISP_BLS },
  411. { .compatible = "mediatek,mt8173-disp-pwm",
  412. .data = (void *)MTK_DISP_PWM },
  413. { .compatible = "mediatek,mt8173-disp-od",
  414. .data = (void *)MTK_DISP_OD },
  415. { }
  416. };
  417. static int mtk_drm_probe(struct platform_device *pdev)
  418. {
  419. struct device *dev = &pdev->dev;
  420. struct mtk_drm_private *private;
  421. struct resource *mem;
  422. struct device_node *node;
  423. struct component_match *match = NULL;
  424. int ret;
  425. int i;
  426. private = devm_kzalloc(dev, sizeof(*private), GFP_KERNEL);
  427. if (!private)
  428. return -ENOMEM;
  429. mutex_init(&private->commit.lock);
  430. INIT_WORK(&private->commit.work, mtk_atomic_work);
  431. private->data = of_device_get_match_data(dev);
  432. mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  433. private->config_regs = devm_ioremap_resource(dev, mem);
  434. if (IS_ERR(private->config_regs)) {
  435. ret = PTR_ERR(private->config_regs);
  436. dev_err(dev, "Failed to ioremap mmsys-config resource: %d\n",
  437. ret);
  438. return ret;
  439. }
  440. /* Iterate over sibling DISP function blocks */
  441. for_each_child_of_node(dev->of_node->parent, node) {
  442. const struct of_device_id *of_id;
  443. enum mtk_ddp_comp_type comp_type;
  444. int comp_id;
  445. of_id = of_match_node(mtk_ddp_comp_dt_ids, node);
  446. if (!of_id)
  447. continue;
  448. if (!of_device_is_available(node)) {
  449. dev_dbg(dev, "Skipping disabled component %pOF\n",
  450. node);
  451. continue;
  452. }
  453. comp_type = (enum mtk_ddp_comp_type)of_id->data;
  454. if (comp_type == MTK_DISP_MUTEX) {
  455. private->mutex_node = of_node_get(node);
  456. continue;
  457. }
  458. comp_id = mtk_ddp_comp_get_id(node, comp_type);
  459. if (comp_id < 0) {
  460. dev_warn(dev, "Skipping unknown component %pOF\n",
  461. node);
  462. continue;
  463. }
  464. private->comp_node[comp_id] = of_node_get(node);
  465. /*
  466. * Currently only the COLOR, OVL, RDMA, DSI, and DPI blocks have
  467. * separate component platform drivers and initialize their own
  468. * DDP component structure. The others are initialized here.
  469. */
  470. if (comp_type == MTK_DISP_COLOR ||
  471. comp_type == MTK_DISP_OVL ||
  472. comp_type == MTK_DISP_RDMA ||
  473. comp_type == MTK_DSI ||
  474. comp_type == MTK_DPI) {
  475. dev_info(dev, "Adding component match for %pOF\n",
  476. node);
  477. drm_of_component_match_add(dev, &match, compare_of,
  478. node);
  479. } else {
  480. struct mtk_ddp_comp *comp;
  481. comp = devm_kzalloc(dev, sizeof(*comp), GFP_KERNEL);
  482. if (!comp) {
  483. ret = -ENOMEM;
  484. of_node_put(node);
  485. goto err_node;
  486. }
  487. ret = mtk_ddp_comp_init(dev, node, comp, comp_id, NULL);
  488. if (ret) {
  489. of_node_put(node);
  490. goto err_node;
  491. }
  492. private->ddp_comp[comp_id] = comp;
  493. }
  494. }
  495. if (!private->mutex_node) {
  496. dev_err(dev, "Failed to find disp-mutex node\n");
  497. ret = -ENODEV;
  498. goto err_node;
  499. }
  500. pm_runtime_enable(dev);
  501. platform_set_drvdata(pdev, private);
  502. ret = component_master_add_with_match(dev, &mtk_drm_ops, match);
  503. if (ret)
  504. goto err_pm;
  505. return 0;
  506. err_pm:
  507. pm_runtime_disable(dev);
  508. err_node:
  509. of_node_put(private->mutex_node);
  510. for (i = 0; i < DDP_COMPONENT_ID_MAX; i++) {
  511. of_node_put(private->comp_node[i]);
  512. if (private->ddp_comp[i]) {
  513. put_device(private->ddp_comp[i]->larb_dev);
  514. private->ddp_comp[i] = NULL;
  515. }
  516. }
  517. return ret;
  518. }
  519. static int mtk_drm_remove(struct platform_device *pdev)
  520. {
  521. struct mtk_drm_private *private = platform_get_drvdata(pdev);
  522. int i;
  523. component_master_del(&pdev->dev, &mtk_drm_ops);
  524. pm_runtime_disable(&pdev->dev);
  525. of_node_put(private->mutex_node);
  526. for (i = 0; i < DDP_COMPONENT_ID_MAX; i++)
  527. of_node_put(private->comp_node[i]);
  528. return 0;
  529. }
  530. #ifdef CONFIG_PM_SLEEP
  531. static int mtk_drm_sys_suspend(struct device *dev)
  532. {
  533. struct mtk_drm_private *private = dev_get_drvdata(dev);
  534. struct drm_device *drm = private->drm;
  535. int ret;
  536. ret = drm_mode_config_helper_suspend(drm);
  537. DRM_DEBUG_DRIVER("mtk_drm_sys_suspend\n");
  538. return ret;
  539. }
  540. static int mtk_drm_sys_resume(struct device *dev)
  541. {
  542. struct mtk_drm_private *private = dev_get_drvdata(dev);
  543. struct drm_device *drm = private->drm;
  544. int ret;
  545. ret = drm_mode_config_helper_resume(drm);
  546. DRM_DEBUG_DRIVER("mtk_drm_sys_resume\n");
  547. return ret;
  548. }
  549. #endif
  550. static SIMPLE_DEV_PM_OPS(mtk_drm_pm_ops, mtk_drm_sys_suspend,
  551. mtk_drm_sys_resume);
  552. static const struct of_device_id mtk_drm_of_ids[] = {
  553. { .compatible = "mediatek,mt2701-mmsys",
  554. .data = &mt2701_mmsys_driver_data},
  555. { .compatible = "mediatek,mt2712-mmsys",
  556. .data = &mt2712_mmsys_driver_data},
  557. { .compatible = "mediatek,mt8173-mmsys",
  558. .data = &mt8173_mmsys_driver_data},
  559. { }
  560. };
  561. static struct platform_driver mtk_drm_platform_driver = {
  562. .probe = mtk_drm_probe,
  563. .remove = mtk_drm_remove,
  564. .driver = {
  565. .name = "mediatek-drm",
  566. .of_match_table = mtk_drm_of_ids,
  567. .pm = &mtk_drm_pm_ops,
  568. },
  569. };
  570. static struct platform_driver * const mtk_drm_drivers[] = {
  571. &mtk_ddp_driver,
  572. &mtk_disp_color_driver,
  573. &mtk_disp_ovl_driver,
  574. &mtk_disp_rdma_driver,
  575. &mtk_dpi_driver,
  576. &mtk_drm_platform_driver,
  577. &mtk_dsi_driver,
  578. &mtk_mipi_tx_driver,
  579. };
  580. static int __init mtk_drm_init(void)
  581. {
  582. return platform_register_drivers(mtk_drm_drivers,
  583. ARRAY_SIZE(mtk_drm_drivers));
  584. }
  585. static void __exit mtk_drm_exit(void)
  586. {
  587. platform_unregister_drivers(mtk_drm_drivers,
  588. ARRAY_SIZE(mtk_drm_drivers));
  589. }
  590. module_init(mtk_drm_init);
  591. module_exit(mtk_drm_exit);
  592. MODULE_AUTHOR("YT SHEN <yt.shen@mediatek.com>");
  593. MODULE_DESCRIPTION("Mediatek SoC DRM driver");
  594. MODULE_LICENSE("GPL v2");