exynos_mipi_dsi.c 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2012 Samsung Electronics
  4. *
  5. * Author: InKi Dae <inki.dae@samsung.com>
  6. * Author: Donghwa Lee <dh09.lee@samsung.com>
  7. */
  8. #include <common.h>
  9. #include <malloc.h>
  10. #include <fdtdec.h>
  11. #include <linux/libfdt.h>
  12. #include <linux/compat.h>
  13. #include <linux/err.h>
  14. #include <asm/arch/dsim.h>
  15. #include <asm/arch/mipi_dsim.h>
  16. #include <asm/arch/power.h>
  17. #include <asm/arch/cpu.h>
  18. #include <asm/arch/clk.h>
  19. #include "exynos_mipi_dsi_lowlevel.h"
  20. #include "exynos_mipi_dsi_common.h"
  21. #define master_to_driver(a) (a->dsim_lcd_drv)
  22. #define master_to_device(a) (a->dsim_lcd_dev)
  23. DECLARE_GLOBAL_DATA_PTR;
  24. struct mipi_dsim_ddi {
  25. int bus_id;
  26. struct list_head list;
  27. struct mipi_dsim_lcd_device *dsim_lcd_dev;
  28. struct mipi_dsim_lcd_driver *dsim_lcd_drv;
  29. };
  30. static LIST_HEAD(dsim_ddi_list);
  31. static LIST_HEAD(dsim_lcd_dev_list);
  32. int exynos_mipi_dsi_register_lcd_device(struct mipi_dsim_lcd_device *lcd_dev)
  33. {
  34. struct mipi_dsim_ddi *dsim_ddi;
  35. if (!lcd_dev) {
  36. debug("mipi_dsim_lcd_device is NULL.\n");
  37. return -EFAULT;
  38. }
  39. if (!lcd_dev->name) {
  40. debug("dsim_lcd_device name is NULL.\n");
  41. return -EFAULT;
  42. }
  43. dsim_ddi = kzalloc(sizeof(struct mipi_dsim_ddi), GFP_KERNEL);
  44. if (!dsim_ddi) {
  45. debug("failed to allocate dsim_ddi object.\n");
  46. return -EFAULT;
  47. }
  48. dsim_ddi->dsim_lcd_dev = lcd_dev;
  49. list_add_tail(&dsim_ddi->list, &dsim_ddi_list);
  50. return 0;
  51. }
  52. struct mipi_dsim_ddi
  53. *exynos_mipi_dsi_find_lcd_device(struct mipi_dsim_lcd_driver *lcd_drv)
  54. {
  55. struct mipi_dsim_ddi *dsim_ddi;
  56. struct mipi_dsim_lcd_device *lcd_dev;
  57. list_for_each_entry(dsim_ddi, &dsim_ddi_list, list) {
  58. lcd_dev = dsim_ddi->dsim_lcd_dev;
  59. if (!lcd_dev)
  60. continue;
  61. if (lcd_drv->id >= 0) {
  62. if ((strcmp(lcd_drv->name, lcd_dev->name)) == 0 &&
  63. lcd_drv->id == lcd_dev->id) {
  64. /**
  65. * bus_id would be used to identify
  66. * connected bus.
  67. */
  68. dsim_ddi->bus_id = lcd_dev->bus_id;
  69. return dsim_ddi;
  70. }
  71. } else {
  72. if ((strcmp(lcd_drv->name, lcd_dev->name)) == 0) {
  73. /**
  74. * bus_id would be used to identify
  75. * connected bus.
  76. */
  77. dsim_ddi->bus_id = lcd_dev->bus_id;
  78. return dsim_ddi;
  79. }
  80. }
  81. kfree(dsim_ddi);
  82. list_del(&dsim_ddi_list);
  83. }
  84. return NULL;
  85. }
  86. int exynos_mipi_dsi_register_lcd_driver(struct mipi_dsim_lcd_driver *lcd_drv)
  87. {
  88. struct mipi_dsim_ddi *dsim_ddi;
  89. if (!lcd_drv) {
  90. debug("mipi_dsim_lcd_driver is NULL.\n");
  91. return -EFAULT;
  92. }
  93. if (!lcd_drv->name) {
  94. debug("dsim_lcd_driver name is NULL.\n");
  95. return -EFAULT;
  96. }
  97. dsim_ddi = exynos_mipi_dsi_find_lcd_device(lcd_drv);
  98. if (!dsim_ddi) {
  99. debug("mipi_dsim_ddi object not found.\n");
  100. return -EFAULT;
  101. }
  102. dsim_ddi->dsim_lcd_drv = lcd_drv;
  103. debug("registered panel driver(%s) to mipi-dsi driver.\n",
  104. lcd_drv->name);
  105. return 0;
  106. }
  107. struct mipi_dsim_ddi
  108. *exynos_mipi_dsi_bind_lcd_ddi(struct mipi_dsim_device *dsim,
  109. const char *name)
  110. {
  111. struct mipi_dsim_ddi *dsim_ddi;
  112. struct mipi_dsim_lcd_driver *lcd_drv;
  113. struct mipi_dsim_lcd_device *lcd_dev;
  114. list_for_each_entry(dsim_ddi, &dsim_ddi_list, list) {
  115. lcd_drv = dsim_ddi->dsim_lcd_drv;
  116. lcd_dev = dsim_ddi->dsim_lcd_dev;
  117. if (!lcd_drv || !lcd_dev)
  118. continue;
  119. debug("lcd_drv->id = %d, lcd_dev->id = %d\n",
  120. lcd_drv->id, lcd_dev->id);
  121. if ((strcmp(lcd_drv->name, name) == 0)) {
  122. lcd_dev->master = dsim;
  123. dsim->dsim_lcd_dev = lcd_dev;
  124. dsim->dsim_lcd_drv = lcd_drv;
  125. return dsim_ddi;
  126. }
  127. }
  128. return NULL;
  129. }
  130. /* define MIPI-DSI Master operations. */
  131. static struct mipi_dsim_master_ops master_ops = {
  132. .cmd_write = exynos_mipi_dsi_wr_data,
  133. .get_dsim_frame_done = exynos_mipi_dsi_get_frame_done_status,
  134. .clear_dsim_frame_done = exynos_mipi_dsi_clear_frame_done,
  135. };
  136. int exynos_mipi_dsi_init(struct exynos_platform_mipi_dsim *dsim_pd)
  137. {
  138. struct mipi_dsim_device *dsim;
  139. struct mipi_dsim_config *dsim_config;
  140. struct mipi_dsim_ddi *dsim_ddi;
  141. dsim = kzalloc(sizeof(struct mipi_dsim_device), GFP_KERNEL);
  142. if (!dsim) {
  143. debug("failed to allocate dsim object.\n");
  144. return -EFAULT;
  145. }
  146. /* get mipi_dsim_config. */
  147. dsim_config = dsim_pd->dsim_config;
  148. if (dsim_config == NULL) {
  149. debug("failed to get dsim config data.\n");
  150. return -EFAULT;
  151. }
  152. dsim->pd = dsim_pd;
  153. dsim->dsim_config = dsim_config;
  154. dsim->master_ops = &master_ops;
  155. /* bind lcd ddi matched with panel name. */
  156. dsim_ddi = exynos_mipi_dsi_bind_lcd_ddi(dsim, dsim_pd->lcd_panel_name);
  157. if (!dsim_ddi) {
  158. debug("mipi_dsim_ddi object not found.\n");
  159. return -ENOSYS;
  160. }
  161. if (dsim_pd->lcd_power)
  162. dsim_pd->lcd_power();
  163. if (dsim_pd->mipi_power)
  164. dsim_pd->mipi_power();
  165. /* phy_enable(unsigned int dev_index, unsigned int enable) */
  166. if (dsim_pd->phy_enable)
  167. dsim_pd->phy_enable(0, 1);
  168. set_mipi_clk();
  169. exynos_mipi_dsi_init_dsim(dsim);
  170. exynos_mipi_dsi_init_link(dsim);
  171. exynos_mipi_dsi_set_hs_enable(dsim);
  172. /* set display timing. */
  173. exynos_mipi_dsi_set_display_mode(dsim, dsim->dsim_config);
  174. /* initialize mipi-dsi client(lcd panel). */
  175. if (dsim_ddi->dsim_lcd_drv && dsim_ddi->dsim_lcd_drv->mipi_panel_init) {
  176. dsim_ddi->dsim_lcd_drv->mipi_panel_init(dsim);
  177. dsim_ddi->dsim_lcd_drv->mipi_display_on(dsim);
  178. }
  179. debug("mipi-dsi driver(%s mode) has been probed.\n",
  180. (dsim_config->e_interface == DSIM_COMMAND) ?
  181. "CPU" : "RGB");
  182. return 0;
  183. }
  184. int exynos_dsim_config_parse_dt(const void *blob, struct mipi_dsim_config *dt,
  185. struct mipi_dsim_lcd_device *lcd_dt)
  186. {
  187. int node;
  188. node = fdtdec_next_compatible(blob, 0, COMPAT_SAMSUNG_EXYNOS_MIPI_DSI);
  189. if (node <= 0) {
  190. printf("exynos_mipi_dsi: Can't get device node for mipi dsi\n");
  191. return -ENODEV;
  192. }
  193. dt->e_interface = fdtdec_get_int(blob, node,
  194. "samsung,dsim-config-e-interface", 0);
  195. dt->e_virtual_ch = fdtdec_get_int(blob, node,
  196. "samsung,dsim-config-e-virtual-ch", 0);
  197. dt->e_pixel_format = fdtdec_get_int(blob, node,
  198. "samsung,dsim-config-e-pixel-format", 0);
  199. dt->e_burst_mode = fdtdec_get_int(blob, node,
  200. "samsung,dsim-config-e-burst-mode", 0);
  201. dt->e_no_data_lane = fdtdec_get_int(blob, node,
  202. "samsung,dsim-config-e-no-data-lane", 0);
  203. dt->e_byte_clk = fdtdec_get_int(blob, node,
  204. "samsung,dsim-config-e-byte-clk", 0);
  205. dt->hfp = fdtdec_get_int(blob, node,
  206. "samsung,dsim-config-hfp", 0);
  207. dt->p = fdtdec_get_int(blob, node,
  208. "samsung,dsim-config-p", 0);
  209. dt->m = fdtdec_get_int(blob, node,
  210. "samsung,dsim-config-m", 0);
  211. dt->s = fdtdec_get_int(blob, node,
  212. "samsung,dsim-config-s", 0);
  213. dt->pll_stable_time = fdtdec_get_int(blob, node,
  214. "samsung,dsim-config-pll-stable-time", 0);
  215. dt->esc_clk = fdtdec_get_int(blob, node,
  216. "samsung,dsim-config-esc-clk", 0);
  217. dt->stop_holding_cnt = fdtdec_get_int(blob, node,
  218. "samsung,dsim-config-stop-holding-cnt", 0);
  219. dt->bta_timeout = fdtdec_get_int(blob, node,
  220. "samsung,dsim-config-bta-timeout", 0);
  221. dt->rx_timeout = fdtdec_get_int(blob, node,
  222. "samsung,dsim-config-rx-timeout", 0);
  223. lcd_dt->name = fdtdec_get_config_string(blob,
  224. "samsung,dsim-device-name");
  225. lcd_dt->id = fdtdec_get_int(blob, node,
  226. "samsung,dsim-device-id", 0);
  227. lcd_dt->bus_id = fdtdec_get_int(blob, node,
  228. "samsung,dsim-device-bus_id", 0);
  229. lcd_dt->reverse_panel = fdtdec_get_int(blob, node,
  230. "samsung,dsim-device-reverse-panel", 0);
  231. return 0;
  232. }
  233. void exynos_init_dsim_platform_data(vidinfo_t *vid)
  234. {
  235. static struct mipi_dsim_config dsim_config_dt;
  236. static struct exynos_platform_mipi_dsim dsim_platform_data_dt;
  237. static struct mipi_dsim_lcd_device mipi_lcd_device_dt;
  238. if (exynos_dsim_config_parse_dt(gd->fdt_blob, &dsim_config_dt,
  239. &mipi_lcd_device_dt))
  240. debug("Can't get proper dsim config.\n");
  241. strcpy(dsim_platform_data_dt.lcd_panel_name, mipi_lcd_device_dt.name);
  242. dsim_platform_data_dt.dsim_config = &dsim_config_dt;
  243. dsim_platform_data_dt.mipi_power = mipi_power;
  244. dsim_platform_data_dt.phy_enable = set_mipi_phy_ctrl;
  245. dsim_platform_data_dt.lcd_panel_info = (void *)vid;
  246. mipi_lcd_device_dt.platform_data = (void *)&dsim_platform_data_dt;
  247. exynos_mipi_dsi_register_lcd_device(&mipi_lcd_device_dt);
  248. vid->dsim_platform_data_dt = &dsim_platform_data_dt;
  249. }