clap070wp03xg_lvds_8163.c 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. #ifdef BUILD_LK
  2. #include <platform/mt_gpio.h>
  3. #include <platform/mt_i2c.h>
  4. #include <platform/mt_pmic.h>
  5. #include <string.h>
  6. #else
  7. #include <linux/string.h>
  8. #include <linux/wait.h>
  9. #include <linux/platform_device.h>
  10. #include <linux/gpio.h>
  11. #include <linux/pinctrl/consumer.h>
  12. #include <linux/of_gpio.h>
  13. #include <asm-generic/gpio.h>
  14. #include <linux/kernel.h>
  15. #include <linux/mm.h>
  16. #include <linux/mm_types.h>
  17. #include <linux/module.h>
  18. #include <linux/types.h>
  19. #include <linux/slab.h>
  20. #include <linux/vmalloc.h>
  21. #ifdef CONFIG_OF
  22. #include <linux/of.h>
  23. #include <linux/of_irq.h>
  24. #include <linux/of_address.h>
  25. #include <linux/of_device.h>
  26. #include <linux/regulator/consumer.h>
  27. #include <linux/clk.h>
  28. #endif
  29. #endif
  30. #include "lcm_drv.h"
  31. /*#include "ddp_irq.h"*/
  32. #ifdef BUILD_LK
  33. #ifdef GPIO_LCM_PWR
  34. #define GPIO_LCD_PWR GPIO_LCM_PWR
  35. #else
  36. #define GPIO_LCD_PWR 0xFFFFFFFF
  37. #endif
  38. static void lcm_set_gpio_output(unsigned int GPIO, unsigned int output)
  39. {
  40. mt_set_gpio_mode(GPIO, GPIO_MODE_00);
  41. mt_set_gpio_dir(GPIO, GPIO_DIR_OUT);
  42. mt_set_gpio_out(GPIO, (output > 0) ? GPIO_OUT_ONE : GPIO_OUT_ZERO);
  43. }
  44. #else
  45. /*static unsigned int GPIO_LCD_PWR_EN;*/
  46. static struct regulator *lcm_vgp;
  47. static struct pinctrl *lcmctrl;
  48. static struct pinctrl_state *lcd_pwr_high;
  49. static struct pinctrl_state *lcd_pwr_low;
  50. static int lcm_get_gpio(struct device *dev)
  51. {
  52. int ret = 0;
  53. lcmctrl = devm_pinctrl_get(dev);
  54. if (IS_ERR(lcmctrl)) {
  55. dev_err(dev, "Cannot find lcm pinctrl!");
  56. ret = PTR_ERR(lcmctrl);
  57. }
  58. /*lcm power pin lookup */
  59. lcd_pwr_high = pinctrl_lookup_state(lcmctrl, "lcm_pwr_high");
  60. if (IS_ERR(lcd_pwr_high)) {
  61. ret = PTR_ERR(lcd_pwr_high);
  62. pr_debug("%s : pinctrl err, lcd_pwr_high\n", __func__);
  63. }
  64. lcd_pwr_low = pinctrl_lookup_state(lcmctrl, "lcm_pwr_low");
  65. if (IS_ERR(lcd_pwr_low)) {
  66. ret = PTR_ERR(lcd_pwr_low);
  67. pr_debug("%s : pinctrl err, lcd_pwr_low\n", __func__);
  68. }
  69. return ret;
  70. }
  71. void lcm_set_gpio(int val)
  72. {
  73. if (val == 0) {
  74. pinctrl_select_state(lcmctrl, lcd_pwr_low);
  75. pr_debug("LCM: lcm set power off\n");
  76. } else {
  77. pinctrl_select_state(lcmctrl, lcd_pwr_high);
  78. pr_debug("LCM: lcm set power on\n");
  79. }
  80. }
  81. /* get LDO supply */
  82. static int lcm_get_vgp_supply(struct device *dev)
  83. {
  84. int ret;
  85. struct regulator *lcm_vgp_ldo;
  86. pr_debug("LCM: lcm_get_vgp_supply is going\n");
  87. lcm_vgp_ldo = devm_regulator_get(dev, "reg-lcm");
  88. if (IS_ERR(lcm_vgp_ldo)) {
  89. ret = PTR_ERR(lcm_vgp_ldo);
  90. dev_err(dev, "failed to get reg-lcm LDO, %d\n", ret);
  91. return ret;
  92. }
  93. pr_debug("LCM: lcm get supply ok.\n");
  94. ret = regulator_enable(lcm_vgp_ldo);
  95. /* get current voltage settings */
  96. ret = regulator_get_voltage(lcm_vgp_ldo);
  97. pr_debug("lcm LDO voltage = %d in LK stage\n", ret);
  98. lcm_vgp = lcm_vgp_ldo;
  99. return ret;
  100. }
  101. int lcm_vgp_supply_enable(void)
  102. {
  103. int ret;
  104. unsigned int volt;
  105. pr_debug("LCM: lcm_vgp_supply_enable\n");
  106. if (NULL == lcm_vgp)
  107. return 0;
  108. pr_debug("LCM: set regulator voltage lcm_vgp voltage to 1.8V\n");
  109. /* set voltage to 1.8V */
  110. ret = regulator_set_voltage(lcm_vgp, 1800000, 1800000);
  111. if (ret != 0) {
  112. pr_err("LCM: lcm failed to set lcm_vgp voltage: %d\n", ret);
  113. return ret;
  114. }
  115. /* get voltage settings again */
  116. volt = regulator_get_voltage(lcm_vgp);
  117. if (volt == 1800000)
  118. pr_err("LCM: check regulator voltage=1800000 pass!\n");
  119. else
  120. pr_err("LCM: check regulator voltage=1800000 fail! (voltage: %d)\n", volt);
  121. ret = regulator_enable(lcm_vgp);
  122. if (ret != 0) {
  123. pr_err("LCM: Failed to enable lcm_vgp: %d\n", ret);
  124. return ret;
  125. }
  126. return ret;
  127. }
  128. int lcm_vgp_supply_disable(void)
  129. {
  130. int ret = 0;
  131. unsigned int isenable;
  132. if (NULL == lcm_vgp)
  133. return 0;
  134. /* disable regulator */
  135. isenable = regulator_is_enabled(lcm_vgp);
  136. pr_debug("LCM: lcm query regulator enable status[0x%d]\n", isenable);
  137. if (isenable) {
  138. ret = regulator_disable(lcm_vgp);
  139. if (ret != 0) {
  140. pr_err("LCM: lcm failed to disable lcm_vgp: %d\n", ret);
  141. return ret;
  142. }
  143. /* verify */
  144. isenable = regulator_is_enabled(lcm_vgp);
  145. if (!isenable)
  146. pr_err("LCM: lcm regulator disable pass\n");
  147. }
  148. return ret;
  149. }
  150. static int lcm_probe(struct device *dev)
  151. {
  152. lcm_get_vgp_supply(dev);
  153. lcm_get_gpio(dev);
  154. return 0;
  155. }
  156. static const struct of_device_id lcm_of_ids[] = {
  157. {.compatible = "mediatek,lcm",},
  158. {}
  159. };
  160. static struct platform_driver lcm_driver = {
  161. .driver = {
  162. .name = "mtk_lcm",
  163. .owner = THIS_MODULE,
  164. .probe = lcm_probe,
  165. #ifdef CONFIG_OF
  166. .of_match_table = lcm_of_ids,
  167. #endif
  168. },
  169. };
  170. static int __init lcm_init(void)
  171. {
  172. pr_notice("LCM: Register lcm driver\n");
  173. if (platform_driver_register(&lcm_driver)) {
  174. pr_err("LCM: failed to register disp driver\n");
  175. return -ENODEV;
  176. }
  177. return 0;
  178. }
  179. static void __exit lcm_exit(void)
  180. {
  181. platform_driver_unregister(&lcm_driver);
  182. pr_notice("LCM: Unregister lcm driver done\n");
  183. }
  184. late_initcall(lcm_init);
  185. module_exit(lcm_exit);
  186. MODULE_AUTHOR("mediatek");
  187. MODULE_DESCRIPTION("Display subsystem Driver");
  188. MODULE_LICENSE("GPL");
  189. #endif
  190. /* --------------------------------------------------------------------------- */
  191. /* Local Constants */
  192. /* --------------------------------------------------------------------------- */
  193. #define FRAME_WIDTH (800)
  194. #define FRAME_HEIGHT (1280)
  195. #define HSYNC_PULSE_WIDTH 16
  196. #define HSYNC_BACK_PORCH 16
  197. #define HSYNC_FRONT_PORCH 32
  198. #define VSYNC_PULSE_WIDTH 2
  199. #define VSYNC_BACK_PORCH 2
  200. #define VSYNC_FRONT_PORCH 4
  201. /* --------------------------------------------------------------------------- */
  202. /* Local Variables */
  203. /* --------------------------------------------------------------------------- */
  204. static LCM_UTIL_FUNCS lcm_util = { 0 };
  205. #define SET_RESET_PIN(v) (lcm_util.set_reset_pin((v)))
  206. #define UDELAY(n) (lcm_util.udelay(n))
  207. #define MDELAY(n) (lcm_util.mdelay(n))
  208. /* --------------------------------------------------------------------------- */
  209. /* Local Functions */
  210. /* --------------------------------------------------------------------------- */
  211. static void lcm_init_power(void)
  212. {
  213. #ifdef BUILD_LK
  214. printf("[LK/LCM] lcm_init_power() enter\n");
  215. lcm_set_gpio_output(GPIO_LCD_PWR, GPIO_OUT_ONE);
  216. MDELAY(20);
  217. upmu_set_rg_vgp1_vosel(3);
  218. upmu_set_rg_vgp1_en(0x1);
  219. #else
  220. pr_err("[Kernel/LCM] lcm_init_power() enter\n");
  221. #endif
  222. }
  223. static void lcm_suspend_power(void)
  224. {
  225. #ifdef BUILD_LK
  226. printf("[LK/LCM] lcm_suspend_power() enter\n");
  227. lcm_set_gpio_output(GPIO_LCD_PWR, GPIO_OUT_ZERO);
  228. MDELAY(20);
  229. upmu_set_rg_vgp1_vosel(0);
  230. upmu_set_rg_vgp1_en(0);
  231. #else
  232. pr_err("[Kernel/LCM] lcm_suspend_power() enter\n");
  233. lcm_set_gpio(0);
  234. MDELAY(20);
  235. lcm_vgp_supply_disable();
  236. MDELAY(20);
  237. #endif
  238. }
  239. static void lcm_resume_power(void)
  240. {
  241. #ifdef BUILD_LK
  242. printf("[LK/LCM] lcm_resume_power() enter\n");
  243. lcm_set_gpio_output(GPIO_LCD_PWR, GPIO_OUT_ONE);
  244. MDELAY(20);
  245. upmu_set_rg_vgp1_vosel(3);
  246. upmu_set_rg_vgp1_en(0x1);
  247. #else
  248. pr_err("[Kernel/LCM] lcm_resume_power() enter\n");
  249. lcm_set_gpio(1);
  250. MDELAY(20);
  251. lcm_vgp_supply_enable();
  252. MDELAY(20);
  253. #endif
  254. }
  255. /* --------------------------------------------------------------------------- */
  256. /* LCM Driver Implementations */
  257. /* --------------------------------------------------------------------------- */
  258. static void lcm_set_util_funcs(const LCM_UTIL_FUNCS *util)
  259. {
  260. memcpy(&lcm_util, util, sizeof(LCM_UTIL_FUNCS));
  261. }
  262. static void lcm_get_params(LCM_PARAMS *params)
  263. {
  264. memset(params, 0, sizeof(LCM_PARAMS));
  265. params->type = LCM_TYPE_DPI;
  266. params->width = FRAME_WIDTH;
  267. params->height = FRAME_HEIGHT;
  268. params->dpi.PLL_CLOCK = 72;
  269. params->dpi.width = FRAME_WIDTH;
  270. params->dpi.height = FRAME_HEIGHT;
  271. params->dpi.clk_pol = LCM_POLARITY_FALLING;
  272. params->dpi.de_pol = LCM_POLARITY_RISING;
  273. params->dpi.vsync_pol = LCM_POLARITY_FALLING;
  274. params->dpi.hsync_pol = LCM_POLARITY_FALLING;
  275. params->dpi.hsync_pulse_width = HSYNC_PULSE_WIDTH;
  276. params->dpi.hsync_back_porch = HSYNC_BACK_PORCH;
  277. params->dpi.hsync_front_porch = HSYNC_FRONT_PORCH;
  278. params->dpi.vsync_pulse_width = VSYNC_PULSE_WIDTH;
  279. params->dpi.vsync_back_porch = VSYNC_BACK_PORCH;
  280. params->dpi.vsync_front_porch = VSYNC_FRONT_PORCH;
  281. params->dpi.lvds_tx_en = 1;
  282. params->dpi.format = LCM_DPI_FORMAT_RGB888;
  283. params->dpi.rgb_order = LCM_COLOR_ORDER_RGB;
  284. }
  285. static void lcm_init_lcm(void)
  286. {
  287. #ifdef BUILD_LK
  288. printf("[LK/LCM] lcm_init() enter\n");
  289. SET_RESET_PIN(1);
  290. MDELAY(20);
  291. SET_RESET_PIN(0);
  292. MDELAY(20);
  293. SET_RESET_PIN(1);
  294. MDELAY(20);
  295. #else
  296. pr_err("[Kernel/LCM] lcm_init() enter\n");
  297. #endif
  298. }
  299. void lcm_suspend(void)
  300. {
  301. #ifdef BUILD_LK
  302. printf("[LK/LCM] lcm_suspend() enter\n");
  303. SET_RESET_PIN(1);
  304. MDELAY(10);
  305. SET_RESET_PIN(0);
  306. MDELAY(10);
  307. #else
  308. pr_err("[Kernel/LCM] lcm_suspend() enter\n");
  309. SET_RESET_PIN(1);
  310. MDELAY(10);
  311. SET_RESET_PIN(0);
  312. MDELAY(10);
  313. #endif
  314. }
  315. void lcm_resume(void)
  316. {
  317. #ifdef BUILD_LK
  318. printf("[LK/LCM] lcm_resume() enter\n");
  319. SET_RESET_PIN(1);
  320. MDELAY(20);
  321. SET_RESET_PIN(0);
  322. MDELAY(20);
  323. SET_RESET_PIN(1);
  324. MDELAY(20);
  325. #else
  326. pr_err("[Kernel/LCM] lcm_resume() enter\n");
  327. SET_RESET_PIN(1);
  328. MDELAY(20);
  329. SET_RESET_PIN(0);
  330. MDELAY(20);
  331. SET_RESET_PIN(1);
  332. MDELAY(20);
  333. #endif
  334. }
  335. LCM_DRIVER clap070wp03xg_lvds_8163_lcm_drv = {
  336. .name = "clap070wp03xg_lvds_8163",
  337. .set_util_funcs = lcm_set_util_funcs,
  338. .get_params = lcm_get_params,
  339. .init = lcm_init_lcm,
  340. .suspend = lcm_suspend,
  341. .resume = lcm_resume,
  342. .init_power = lcm_init_power,
  343. .resume_power = lcm_resume_power,
  344. .suspend_power = lcm_suspend_power,
  345. };