exynos_mipi_dsi.c 8.2 KB

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