panel-raydium-rm68200.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) STMicroelectronics SA 2017
  4. *
  5. * Authors: Philippe Cornu <philippe.cornu@st.com>
  6. * Yannick Fertre <yannick.fertre@st.com>
  7. */
  8. #include <linux/delay.h>
  9. #include <linux/gpio/consumer.h>
  10. #include <linux/mod_devicetable.h>
  11. #include <linux/module.h>
  12. #include <linux/regulator/consumer.h>
  13. #include <video/mipi_display.h>
  14. #include <drm/drm_mipi_dsi.h>
  15. #include <drm/drm_modes.h>
  16. #include <drm/drm_panel.h>
  17. /*** Manufacturer Command Set ***/
  18. #define MCS_CMD_MODE_SW 0xFE /* CMD Mode Switch */
  19. #define MCS_CMD1_UCS 0x00 /* User Command Set (UCS = CMD1) */
  20. #define MCS_CMD2_P0 0x01 /* Manufacture Command Set Page0 (CMD2 P0) */
  21. #define MCS_CMD2_P1 0x02 /* Manufacture Command Set Page1 (CMD2 P1) */
  22. #define MCS_CMD2_P2 0x03 /* Manufacture Command Set Page2 (CMD2 P2) */
  23. #define MCS_CMD2_P3 0x04 /* Manufacture Command Set Page3 (CMD2 P3) */
  24. /* CMD2 P0 commands (Display Options and Power) */
  25. #define MCS_STBCTR 0x12 /* TE1 Output Setting Zig-Zag Connection */
  26. #define MCS_SGOPCTR 0x16 /* Source Bias Current */
  27. #define MCS_SDCTR 0x1A /* Source Output Delay Time */
  28. #define MCS_INVCTR 0x1B /* Inversion Type */
  29. #define MCS_EXT_PWR_IC 0x24 /* External PWR IC Control */
  30. #define MCS_SETAVDD 0x27 /* PFM Control for AVDD Output */
  31. #define MCS_SETAVEE 0x29 /* PFM Control for AVEE Output */
  32. #define MCS_BT2CTR 0x2B /* DDVDL Charge Pump Control */
  33. #define MCS_BT3CTR 0x2F /* VGH Charge Pump Control */
  34. #define MCS_BT4CTR 0x34 /* VGL Charge Pump Control */
  35. #define MCS_VCMCTR 0x46 /* VCOM Output Level Control */
  36. #define MCS_SETVGN 0x52 /* VG M/S N Control */
  37. #define MCS_SETVGP 0x54 /* VG M/S P Control */
  38. #define MCS_SW_CTRL 0x5F /* Interface Control for PFM and MIPI */
  39. /* CMD2 P2 commands (GOA Timing Control) - no description in datasheet */
  40. #define GOA_VSTV1 0x00
  41. #define GOA_VSTV2 0x07
  42. #define GOA_VCLK1 0x0E
  43. #define GOA_VCLK2 0x17
  44. #define GOA_VCLK_OPT1 0x20
  45. #define GOA_BICLK1 0x2A
  46. #define GOA_BICLK2 0x37
  47. #define GOA_BICLK3 0x44
  48. #define GOA_BICLK4 0x4F
  49. #define GOA_BICLK_OPT1 0x5B
  50. #define GOA_BICLK_OPT2 0x60
  51. #define MCS_GOA_GPO1 0x6D
  52. #define MCS_GOA_GPO2 0x71
  53. #define MCS_GOA_EQ 0x74
  54. #define MCS_GOA_CLK_GALLON 0x7C
  55. #define MCS_GOA_FS_SEL0 0x7E
  56. #define MCS_GOA_FS_SEL1 0x87
  57. #define MCS_GOA_FS_SEL2 0x91
  58. #define MCS_GOA_FS_SEL3 0x9B
  59. #define MCS_GOA_BS_SEL0 0xAC
  60. #define MCS_GOA_BS_SEL1 0xB5
  61. #define MCS_GOA_BS_SEL2 0xBF
  62. #define MCS_GOA_BS_SEL3 0xC9
  63. #define MCS_GOA_BS_SEL4 0xD3
  64. /* CMD2 P3 commands (Gamma) */
  65. #define MCS_GAMMA_VP 0x60 /* Gamma VP1~VP16 */
  66. #define MCS_GAMMA_VN 0x70 /* Gamma VN1~VN16 */
  67. struct rm68200 {
  68. struct device *dev;
  69. struct drm_panel panel;
  70. struct gpio_desc *reset_gpio;
  71. struct regulator *supply;
  72. };
  73. static const struct drm_display_mode default_mode = {
  74. .clock = 54000,
  75. .hdisplay = 720,
  76. .hsync_start = 720 + 48,
  77. .hsync_end = 720 + 48 + 9,
  78. .htotal = 720 + 48 + 9 + 48,
  79. .vdisplay = 1280,
  80. .vsync_start = 1280 + 12,
  81. .vsync_end = 1280 + 12 + 5,
  82. .vtotal = 1280 + 12 + 5 + 12,
  83. .flags = 0,
  84. .width_mm = 68,
  85. .height_mm = 122,
  86. };
  87. static inline struct rm68200 *panel_to_rm68200(struct drm_panel *panel)
  88. {
  89. return container_of(panel, struct rm68200, panel);
  90. }
  91. static void rm68200_dcs_write_buf(struct rm68200 *ctx, const void *data,
  92. size_t len)
  93. {
  94. struct mipi_dsi_device *dsi = to_mipi_dsi_device(ctx->dev);
  95. int err;
  96. err = mipi_dsi_dcs_write_buffer(dsi, data, len);
  97. if (err < 0)
  98. dev_err_ratelimited(ctx->dev, "MIPI DSI DCS write buffer failed: %d\n", err);
  99. }
  100. static void rm68200_dcs_write_cmd(struct rm68200 *ctx, u8 cmd, u8 value)
  101. {
  102. struct mipi_dsi_device *dsi = to_mipi_dsi_device(ctx->dev);
  103. int err;
  104. err = mipi_dsi_dcs_write(dsi, cmd, &value, 1);
  105. if (err < 0)
  106. dev_err_ratelimited(ctx->dev, "MIPI DSI DCS write failed: %d\n", err);
  107. }
  108. #define dcs_write_seq(ctx, seq...) \
  109. ({ \
  110. static const u8 d[] = { seq }; \
  111. \
  112. rm68200_dcs_write_buf(ctx, d, ARRAY_SIZE(d)); \
  113. })
  114. /*
  115. * This panel is not able to auto-increment all cmd addresses so for some of
  116. * them, we need to send them one by one...
  117. */
  118. #define dcs_write_cmd_seq(ctx, cmd, seq...) \
  119. ({ \
  120. static const u8 d[] = { seq }; \
  121. unsigned int i; \
  122. \
  123. for (i = 0; i < ARRAY_SIZE(d) ; i++) \
  124. rm68200_dcs_write_cmd(ctx, cmd + i, d[i]); \
  125. })
  126. static void rm68200_init_sequence(struct rm68200 *ctx)
  127. {
  128. /* Enter CMD2 with page 0 */
  129. dcs_write_seq(ctx, MCS_CMD_MODE_SW, MCS_CMD2_P0);
  130. dcs_write_cmd_seq(ctx, MCS_EXT_PWR_IC, 0xC0, 0x53, 0x00);
  131. dcs_write_seq(ctx, MCS_BT2CTR, 0xE5);
  132. dcs_write_seq(ctx, MCS_SETAVDD, 0x0A);
  133. dcs_write_seq(ctx, MCS_SETAVEE, 0x0A);
  134. dcs_write_seq(ctx, MCS_SGOPCTR, 0x52);
  135. dcs_write_seq(ctx, MCS_BT3CTR, 0x53);
  136. dcs_write_seq(ctx, MCS_BT4CTR, 0x5A);
  137. dcs_write_seq(ctx, MCS_INVCTR, 0x00);
  138. dcs_write_seq(ctx, MCS_STBCTR, 0x0A);
  139. dcs_write_seq(ctx, MCS_SDCTR, 0x06);
  140. dcs_write_seq(ctx, MCS_VCMCTR, 0x56);
  141. dcs_write_seq(ctx, MCS_SETVGN, 0xA0, 0x00);
  142. dcs_write_seq(ctx, MCS_SETVGP, 0xA0, 0x00);
  143. dcs_write_seq(ctx, MCS_SW_CTRL, 0x11); /* 2 data lanes, see doc */
  144. dcs_write_seq(ctx, MCS_CMD_MODE_SW, MCS_CMD2_P2);
  145. dcs_write_seq(ctx, GOA_VSTV1, 0x05);
  146. dcs_write_seq(ctx, 0x02, 0x0B);
  147. dcs_write_seq(ctx, 0x03, 0x0F);
  148. dcs_write_seq(ctx, 0x04, 0x7D, 0x00, 0x50);
  149. dcs_write_cmd_seq(ctx, GOA_VSTV2, 0x05, 0x16, 0x0D, 0x11, 0x7D, 0x00,
  150. 0x50);
  151. dcs_write_cmd_seq(ctx, GOA_VCLK1, 0x07, 0x08, 0x01, 0x02, 0x00, 0x7D,
  152. 0x00, 0x85, 0x08);
  153. dcs_write_cmd_seq(ctx, GOA_VCLK2, 0x03, 0x04, 0x05, 0x06, 0x00, 0x7D,
  154. 0x00, 0x85, 0x08);
  155. dcs_write_seq(ctx, GOA_VCLK_OPT1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  156. 0x00, 0x00, 0x00, 0x00);
  157. dcs_write_cmd_seq(ctx, GOA_BICLK1, 0x07, 0x08);
  158. dcs_write_seq(ctx, 0x2D, 0x01);
  159. dcs_write_seq(ctx, 0x2F, 0x02, 0x00, 0x40, 0x05, 0x08, 0x54, 0x7D,
  160. 0x00);
  161. dcs_write_cmd_seq(ctx, GOA_BICLK2, 0x03, 0x04, 0x05, 0x06, 0x00);
  162. dcs_write_seq(ctx, 0x3D, 0x40);
  163. dcs_write_seq(ctx, 0x3F, 0x05, 0x08, 0x54, 0x7D, 0x00);
  164. dcs_write_seq(ctx, GOA_BICLK3, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  165. 0x00, 0x00, 0x00, 0x00, 0x00);
  166. dcs_write_seq(ctx, GOA_BICLK4, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  167. 0x00, 0x00);
  168. dcs_write_seq(ctx, 0x58, 0x00, 0x00, 0x00);
  169. dcs_write_seq(ctx, GOA_BICLK_OPT1, 0x00, 0x00, 0x00, 0x00, 0x00);
  170. dcs_write_seq(ctx, GOA_BICLK_OPT2, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  171. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00);
  172. dcs_write_seq(ctx, MCS_GOA_GPO1, 0x00, 0x00, 0x00, 0x00);
  173. dcs_write_seq(ctx, MCS_GOA_GPO2, 0x00, 0x20, 0x00);
  174. dcs_write_seq(ctx, MCS_GOA_EQ, 0x08, 0x08, 0x08, 0x08, 0x08, 0x08,
  175. 0x00, 0x00);
  176. dcs_write_seq(ctx, MCS_GOA_CLK_GALLON, 0x00, 0x00);
  177. dcs_write_cmd_seq(ctx, MCS_GOA_FS_SEL0, 0xBF, 0x02, 0x06, 0x14, 0x10,
  178. 0x16, 0x12, 0x08, 0x3F);
  179. dcs_write_cmd_seq(ctx, MCS_GOA_FS_SEL1, 0x3F, 0x3F, 0x3F, 0x3F, 0x0C,
  180. 0x0A, 0x0E, 0x3F, 0x3F, 0x00);
  181. dcs_write_cmd_seq(ctx, MCS_GOA_FS_SEL2, 0x04, 0x3F, 0x3F, 0x3F, 0x3F,
  182. 0x05, 0x01, 0x3F, 0x3F, 0x0F);
  183. dcs_write_cmd_seq(ctx, MCS_GOA_FS_SEL3, 0x0B, 0x0D, 0x3F, 0x3F, 0x3F,
  184. 0x3F);
  185. dcs_write_cmd_seq(ctx, 0xA2, 0x3F, 0x09, 0x13, 0x17, 0x11, 0x15);
  186. dcs_write_cmd_seq(ctx, 0xA9, 0x07, 0x03, 0x3F);
  187. dcs_write_cmd_seq(ctx, MCS_GOA_BS_SEL0, 0x3F, 0x05, 0x01, 0x17, 0x13,
  188. 0x15, 0x11, 0x0F, 0x3F);
  189. dcs_write_cmd_seq(ctx, MCS_GOA_BS_SEL1, 0x3F, 0x3F, 0x3F, 0x3F, 0x0B,
  190. 0x0D, 0x09, 0x3F, 0x3F, 0x07);
  191. dcs_write_cmd_seq(ctx, MCS_GOA_BS_SEL2, 0x03, 0x3F, 0x3F, 0x3F, 0x3F,
  192. 0x02, 0x06, 0x3F, 0x3F, 0x08);
  193. dcs_write_cmd_seq(ctx, MCS_GOA_BS_SEL3, 0x0C, 0x0A, 0x3F, 0x3F, 0x3F,
  194. 0x3F, 0x3F, 0x0E, 0x10, 0x14);
  195. dcs_write_cmd_seq(ctx, MCS_GOA_BS_SEL4, 0x12, 0x16, 0x00, 0x04, 0x3F);
  196. dcs_write_seq(ctx, 0xDC, 0x02);
  197. dcs_write_seq(ctx, 0xDE, 0x12);
  198. dcs_write_seq(ctx, MCS_CMD_MODE_SW, 0x0E); /* No documentation */
  199. dcs_write_seq(ctx, 0x01, 0x75);
  200. dcs_write_seq(ctx, MCS_CMD_MODE_SW, MCS_CMD2_P3);
  201. dcs_write_cmd_seq(ctx, MCS_GAMMA_VP, 0x00, 0x0C, 0x12, 0x0E, 0x06,
  202. 0x12, 0x0E, 0x0B, 0x15, 0x0B, 0x10, 0x07, 0x0F,
  203. 0x12, 0x0C, 0x00);
  204. dcs_write_cmd_seq(ctx, MCS_GAMMA_VN, 0x00, 0x0C, 0x12, 0x0E, 0x06,
  205. 0x12, 0x0E, 0x0B, 0x15, 0x0B, 0x10, 0x07, 0x0F,
  206. 0x12, 0x0C, 0x00);
  207. /* Exit CMD2 */
  208. dcs_write_seq(ctx, MCS_CMD_MODE_SW, MCS_CMD1_UCS);
  209. }
  210. static int rm68200_unprepare(struct drm_panel *panel)
  211. {
  212. struct rm68200 *ctx = panel_to_rm68200(panel);
  213. struct mipi_dsi_device *dsi = to_mipi_dsi_device(ctx->dev);
  214. int ret;
  215. ret = mipi_dsi_dcs_set_display_off(dsi);
  216. if (ret)
  217. dev_warn(panel->dev, "failed to set display off: %d\n", ret);
  218. ret = mipi_dsi_dcs_enter_sleep_mode(dsi);
  219. if (ret)
  220. dev_warn(panel->dev, "failed to enter sleep mode: %d\n", ret);
  221. msleep(120);
  222. if (ctx->reset_gpio) {
  223. gpiod_set_value_cansleep(ctx->reset_gpio, 1);
  224. msleep(20);
  225. }
  226. regulator_disable(ctx->supply);
  227. return 0;
  228. }
  229. static int rm68200_prepare(struct drm_panel *panel)
  230. {
  231. struct rm68200 *ctx = panel_to_rm68200(panel);
  232. struct mipi_dsi_device *dsi = to_mipi_dsi_device(ctx->dev);
  233. int ret;
  234. ret = regulator_enable(ctx->supply);
  235. if (ret < 0) {
  236. dev_err(ctx->dev, "failed to enable supply: %d\n", ret);
  237. return ret;
  238. }
  239. if (ctx->reset_gpio) {
  240. gpiod_set_value_cansleep(ctx->reset_gpio, 1);
  241. msleep(20);
  242. gpiod_set_value_cansleep(ctx->reset_gpio, 0);
  243. msleep(100);
  244. }
  245. rm68200_init_sequence(ctx);
  246. ret = mipi_dsi_dcs_exit_sleep_mode(dsi);
  247. if (ret)
  248. return ret;
  249. msleep(125);
  250. ret = mipi_dsi_dcs_set_display_on(dsi);
  251. if (ret)
  252. return ret;
  253. msleep(20);
  254. return 0;
  255. }
  256. static int rm68200_get_modes(struct drm_panel *panel,
  257. struct drm_connector *connector)
  258. {
  259. struct drm_display_mode *mode;
  260. mode = drm_mode_duplicate(connector->dev, &default_mode);
  261. if (!mode) {
  262. dev_err(panel->dev, "failed to add mode %ux%u@%u\n",
  263. default_mode.hdisplay, default_mode.vdisplay,
  264. drm_mode_vrefresh(&default_mode));
  265. return -ENOMEM;
  266. }
  267. drm_mode_set_name(mode);
  268. mode->type = DRM_MODE_TYPE_DRIVER | DRM_MODE_TYPE_PREFERRED;
  269. drm_mode_probed_add(connector, mode);
  270. connector->display_info.width_mm = mode->width_mm;
  271. connector->display_info.height_mm = mode->height_mm;
  272. return 1;
  273. }
  274. static const struct drm_panel_funcs rm68200_drm_funcs = {
  275. .unprepare = rm68200_unprepare,
  276. .prepare = rm68200_prepare,
  277. .get_modes = rm68200_get_modes,
  278. };
  279. static int rm68200_probe(struct mipi_dsi_device *dsi)
  280. {
  281. struct device *dev = &dsi->dev;
  282. struct rm68200 *ctx;
  283. int ret;
  284. ctx = devm_kzalloc(dev, sizeof(*ctx), GFP_KERNEL);
  285. if (!ctx)
  286. return -ENOMEM;
  287. ctx->reset_gpio = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_LOW);
  288. if (IS_ERR(ctx->reset_gpio)) {
  289. ret = PTR_ERR(ctx->reset_gpio);
  290. dev_err(dev, "cannot get reset GPIO: %d\n", ret);
  291. return ret;
  292. }
  293. ctx->supply = devm_regulator_get(dev, "power");
  294. if (IS_ERR(ctx->supply)) {
  295. ret = PTR_ERR(ctx->supply);
  296. if (ret != -EPROBE_DEFER)
  297. dev_err(dev, "cannot get regulator: %d\n", ret);
  298. return ret;
  299. }
  300. mipi_dsi_set_drvdata(dsi, ctx);
  301. ctx->dev = dev;
  302. dsi->lanes = 2;
  303. dsi->format = MIPI_DSI_FMT_RGB888;
  304. dsi->mode_flags = MIPI_DSI_MODE_VIDEO | MIPI_DSI_MODE_VIDEO_BURST |
  305. MIPI_DSI_MODE_LPM | MIPI_DSI_CLOCK_NON_CONTINUOUS;
  306. drm_panel_init(&ctx->panel, dev, &rm68200_drm_funcs,
  307. DRM_MODE_CONNECTOR_DSI);
  308. ret = drm_panel_of_backlight(&ctx->panel);
  309. if (ret)
  310. return ret;
  311. drm_panel_add(&ctx->panel);
  312. ret = mipi_dsi_attach(dsi);
  313. if (ret < 0) {
  314. dev_err(dev, "mipi_dsi_attach() failed: %d\n", ret);
  315. drm_panel_remove(&ctx->panel);
  316. return ret;
  317. }
  318. return 0;
  319. }
  320. static void rm68200_remove(struct mipi_dsi_device *dsi)
  321. {
  322. struct rm68200 *ctx = mipi_dsi_get_drvdata(dsi);
  323. mipi_dsi_detach(dsi);
  324. drm_panel_remove(&ctx->panel);
  325. }
  326. static const struct of_device_id raydium_rm68200_of_match[] = {
  327. { .compatible = "raydium,rm68200" },
  328. { }
  329. };
  330. MODULE_DEVICE_TABLE(of, raydium_rm68200_of_match);
  331. static struct mipi_dsi_driver raydium_rm68200_driver = {
  332. .probe = rm68200_probe,
  333. .remove = rm68200_remove,
  334. .driver = {
  335. .name = "panel-raydium-rm68200",
  336. .of_match_table = raydium_rm68200_of_match,
  337. },
  338. };
  339. module_mipi_dsi_driver(raydium_rm68200_driver);
  340. MODULE_AUTHOR("Philippe Cornu <philippe.cornu@st.com>");
  341. MODULE_AUTHOR("Yannick Fertre <yannick.fertre@st.com>");
  342. MODULE_DESCRIPTION("DRM Driver for Raydium RM68200 MIPI DSI panel");
  343. MODULE_LICENSE("GPL v2");