#ifdef BUILD_LK #include #include #include #include #else #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef CONFIG_OF #include #include #include #include #include #include #endif #endif #include "lcm_drv.h" /*#include "ddp_irq.h"*/ #ifdef BUILD_LK #ifdef GPIO_LCM_PWR #define GPIO_LCD_PWR GPIO_LCM_PWR #else #define GPIO_LCD_PWR 0xFFFFFFFF #endif static void lcm_set_gpio_output(unsigned int GPIO, unsigned int output) { mt_set_gpio_mode(GPIO, GPIO_MODE_00); mt_set_gpio_dir(GPIO, GPIO_DIR_OUT); mt_set_gpio_out(GPIO, (output > 0) ? GPIO_OUT_ONE : GPIO_OUT_ZERO); } #else /*static unsigned int GPIO_LCD_PWR_EN;*/ static struct regulator *lcm_vgp; static struct pinctrl *lcmctrl; static struct pinctrl_state *lcd_pwr_high; static struct pinctrl_state *lcd_pwr_low; static int lcm_get_gpio(struct device *dev) { int ret = 0; lcmctrl = devm_pinctrl_get(dev); if (IS_ERR(lcmctrl)) { dev_err(dev, "Cannot find lcm pinctrl!"); ret = PTR_ERR(lcmctrl); } /*lcm power pin lookup */ lcd_pwr_high = pinctrl_lookup_state(lcmctrl, "lcm_pwr_high"); if (IS_ERR(lcd_pwr_high)) { ret = PTR_ERR(lcd_pwr_high); pr_debug("%s : pinctrl err, lcd_pwr_high\n", __func__); } lcd_pwr_low = pinctrl_lookup_state(lcmctrl, "lcm_pwr_low"); if (IS_ERR(lcd_pwr_low)) { ret = PTR_ERR(lcd_pwr_low); pr_debug("%s : pinctrl err, lcd_pwr_low\n", __func__); } return ret; } void lcm_set_gpio(int val) { if (val == 0) { pinctrl_select_state(lcmctrl, lcd_pwr_low); pr_debug("LCM: lcm set power off\n"); } else { pinctrl_select_state(lcmctrl, lcd_pwr_high); pr_debug("LCM: lcm set power on\n"); } } /* get LDO supply */ static int lcm_get_vgp_supply(struct device *dev) { int ret; struct regulator *lcm_vgp_ldo; pr_debug("LCM: lcm_get_vgp_supply is going\n"); lcm_vgp_ldo = devm_regulator_get(dev, "reg-lcm"); if (IS_ERR(lcm_vgp_ldo)) { ret = PTR_ERR(lcm_vgp_ldo); dev_err(dev, "failed to get reg-lcm LDO, %d\n", ret); return ret; } pr_debug("LCM: lcm get supply ok.\n"); ret = regulator_enable(lcm_vgp_ldo); /* get current voltage settings */ ret = regulator_get_voltage(lcm_vgp_ldo); pr_debug("lcm LDO voltage = %d in LK stage\n", ret); lcm_vgp = lcm_vgp_ldo; return ret; } int lcm_vgp_supply_enable(void) { int ret; unsigned int volt; pr_debug("LCM: lcm_vgp_supply_enable\n"); if (NULL == lcm_vgp) return 0; pr_debug("LCM: set regulator voltage lcm_vgp voltage to 1.8V\n"); /* set voltage to 1.8V */ ret = regulator_set_voltage(lcm_vgp, 1800000, 1800000); if (ret != 0) { pr_err("LCM: lcm failed to set lcm_vgp voltage: %d\n", ret); return ret; } /* get voltage settings again */ volt = regulator_get_voltage(lcm_vgp); if (volt == 1800000) pr_err("LCM: check regulator voltage=1800000 pass!\n"); else pr_err("LCM: check regulator voltage=1800000 fail! (voltage: %d)\n", volt); ret = regulator_enable(lcm_vgp); if (ret != 0) { pr_err("LCM: Failed to enable lcm_vgp: %d\n", ret); return ret; } return ret; } int lcm_vgp_supply_disable(void) { int ret = 0; unsigned int isenable; if (NULL == lcm_vgp) return 0; /* disable regulator */ isenable = regulator_is_enabled(lcm_vgp); pr_debug("LCM: lcm query regulator enable status[0x%d]\n", isenable); if (isenable) { ret = regulator_disable(lcm_vgp); if (ret != 0) { pr_err("LCM: lcm failed to disable lcm_vgp: %d\n", ret); return ret; } /* verify */ isenable = regulator_is_enabled(lcm_vgp); if (!isenable) pr_err("LCM: lcm regulator disable pass\n"); } return ret; } static int lcm_probe(struct device *dev) { lcm_get_vgp_supply(dev); lcm_get_gpio(dev); return 0; } static const struct of_device_id lcm_of_ids[] = { {.compatible = "mediatek,lcm",}, {} }; static struct platform_driver lcm_driver = { .driver = { .name = "mtk_lcm", .owner = THIS_MODULE, .probe = lcm_probe, #ifdef CONFIG_OF .of_match_table = lcm_of_ids, #endif }, }; static int __init lcm_init(void) { pr_notice("LCM: Register lcm driver\n"); if (platform_driver_register(&lcm_driver)) { pr_err("LCM: failed to register disp driver\n"); return -ENODEV; } return 0; } static void __exit lcm_exit(void) { platform_driver_unregister(&lcm_driver); pr_notice("LCM: Unregister lcm driver done\n"); } late_initcall(lcm_init); module_exit(lcm_exit); MODULE_AUTHOR("mediatek"); MODULE_DESCRIPTION("Display subsystem Driver"); MODULE_LICENSE("GPL"); #endif /* --------------------------------------------------------------------------- */ /* Local Constants */ /* --------------------------------------------------------------------------- */ #define FRAME_WIDTH (800) #define FRAME_HEIGHT (1280) /* --------------------------------------------------------------------------- */ /* Local Variables */ /* --------------------------------------------------------------------------- */ static LCM_UTIL_FUNCS lcm_util = { 0 }; #define SET_RESET_PIN(v) (lcm_util.set_reset_pin((v))) #define UDELAY(n) (lcm_util.udelay(n)) #define MDELAY(n) (lcm_util.mdelay(n)) /* --------------------------------------------------------------------------- */ /* Local Functions */ /* --------------------------------------------------------------------------- */ #define dsi_set_cmdq_V2(cmd, count, ppara, force_update)lcm_util.dsi_set_cmdq_V2(cmd, count, ppara, force_update) #define dsi_set_cmdq(pdata, queue_size, force_update) lcm_util.dsi_set_cmdq(pdata, queue_size, force_update) #define wrtie_cmd(cmd) lcm_util.dsi_write_cmd(cmd) #define write_regs(addr, pdata, byte_nums) lcm_util.dsi_write_regs(addr, pdata, byte_nums) #define read_reg(cmd) lcm_util.dsi_dcs_read_lcm_reg(cmd) #define read_reg_v2(cmd, buffer, buffer_size) lcm_util.dsi_dcs_read_lcm_reg_v2(cmd, buffer, buffer_size) #define LCM_DSI_CMD_MODE 0 static void lcm_init_power(void) { #ifdef BUILD_LK printf("[LK/LCM] lcm_init_power() enter\n"); lcm_set_gpio_output(GPIO_LCD_PWR, GPIO_OUT_ONE); MDELAY(20); upmu_set_rg_vgp3_vosel(3); upmu_set_rg_vgp3_en(0x1); #else pr_debug("[Kernel/LCM] lcm_init_power() enter\n"); #endif } static void lcm_suspend_power(void) { #ifdef BUILD_LK printf("[LK/LCM] lcm_suspend_power() enter\n"); lcm_set_gpio_output(GPIO_LCD_PWR, GPIO_OUT_ZERO); MDELAY(20); upmu_set_rg_vgp3_vosel(0); upmu_set_rg_vgp3_en(0x0); #else pr_debug("[Kernel/LCM] lcm_suspend_power() enter\n"); lcm_set_gpio(0); MDELAY(20); lcm_vgp_supply_disable(); MDELAY(20); #endif } static void lcm_resume_power(void) { #ifdef BUILD_LK printf("[LK/LCM] lcm_resume_power() enter\n"); lcm_set_gpio_output(GPIO_LCD_PWR, GPIO_OUT_ONE); MDELAY(20); upmu_set_rg_vgp3_vosel(3); upmu_set_rg_vgp3_en(0x1); #else pr_debug("[Kernel/LCM] lcm_resume_power() enter\n"); lcm_set_gpio(1); MDELAY(20); lcm_vgp_supply_enable(); MDELAY(20); #endif } /* --------------------------------------------------------------------------- */ /* LCM Driver Implementations */ /* --------------------------------------------------------------------------- */ static void lcm_set_util_funcs(const LCM_UTIL_FUNCS *util) { memcpy(&lcm_util, util, sizeof(LCM_UTIL_FUNCS)); } static void lcm_get_params(LCM_PARAMS *params) { memset(params, 0, sizeof(LCM_PARAMS)); params->type = LCM_TYPE_DSI; params->width = FRAME_WIDTH; params->height = FRAME_HEIGHT; #if (LCM_DSI_CMD_MODE) params->dsi.mode = CMD_MODE; #else params->dsi.mode = SYNC_EVENT_VDO_MODE; #endif /* DSI */ /* Command mode setting */ /* Three lane or Four lane */ params->dsi.LANE_NUM = LCM_FOUR_LANE; /* The following defined the format for data coming from LCD engine. */ params->dsi.data_format.color_order = LCM_COLOR_ORDER_RGB; params->dsi.data_format.trans_seq = LCM_DSI_TRANS_SEQ_MSB_FIRST; params->dsi.data_format.padding = LCM_DSI_PADDING_ON_LSB; params->dsi.data_format.format = LCM_DSI_FORMAT_RGB888; /* Highly depends on LCD driver capability. */ params->dsi.packet_size = 256; /* Video mode setting */ params->dsi.intermediat_buffer_num = 0; params->dsi.PS = LCM_PACKED_PS_24BIT_RGB888; params->dsi.word_count = FRAME_WIDTH * 3; params->dsi.vertical_sync_active = 6; params->dsi.vertical_backporch = 3; params->dsi.vertical_frontporch = 20; params->dsi.vertical_active_line = FRAME_HEIGHT; params->dsi.horizontal_sync_active = 6; params->dsi.horizontal_backporch = 48; params->dsi.horizontal_frontporch = 16; params->dsi.horizontal_active_pixel = FRAME_WIDTH; params->dsi.ssc_disable = 1; params->dsi.PLL_CLOCK = 221; params->dsi.cont_clock = 1; } static void lcm_init_lcm(void) { #ifdef BUILD_LK printf("[LK/LCM] lcm_init() enter\n"); SET_RESET_PIN(1); MDELAY(20); SET_RESET_PIN(0); MDELAY(20); SET_RESET_PIN(1); MDELAY(20); #else pr_debug("[Kernel/LCM] lcm_init() enter\n"); #endif } void lcm_suspend(void) { #ifdef BUILD_LK printf("[LK/LCM] lcm_suspend() enter\n"); SET_RESET_PIN(1); MDELAY(10); SET_RESET_PIN(0); MDELAY(10); #else pr_debug("[Kernel/LCM] lcm_suspend() enter\n"); SET_RESET_PIN(1); MDELAY(10); SET_RESET_PIN(0); MDELAY(10); #endif } void lcm_resume(void) { #ifdef BUILD_LK printf("[LK/LCM] lcm_resume() enter\n"); SET_RESET_PIN(1); MDELAY(20); SET_RESET_PIN(0); MDELAY(20); SET_RESET_PIN(1); MDELAY(20); #else pr_debug("[Kernel/LCM] lcm_resume() enter\n"); SET_RESET_PIN(1); MDELAY(20); SET_RESET_PIN(0); MDELAY(20); SET_RESET_PIN(1); MDELAY(20); #endif } #if (LCM_DSI_CMD_MODE) static void lcm_update(unsigned int x, unsigned int y, unsigned int width, unsigned int height) { unsigned int x0 = x; unsigned int y0 = y; unsigned int x1 = x0 + width - 1; unsigned int y1 = y0 + height - 1; unsigned char x0_MSB = ((x0 >> 8) & 0xFF); unsigned char x0_LSB = (x0 & 0xFF); unsigned char x1_MSB = ((x1 >> 8) & 0xFF); unsigned char x1_LSB = (x1 & 0xFF); unsigned char y0_MSB = ((y0 >> 8) & 0xFF); unsigned char y0_LSB = (y0 & 0xFF); unsigned char y1_MSB = ((y1 >> 8) & 0xFF); unsigned char y1_LSB = (y1 & 0xFF); unsigned int data_array[16]; data_array[0] = 0x00053902; data_array[1] = (x1_MSB << 24) | (x0_LSB << 16) | (x0_MSB << 8) | 0x2a; data_array[2] = (x1_LSB); dsi_set_cmdq(data_array, 3, 1); data_array[0] = 0x00053902; data_array[1] = (y1_MSB << 24) | (y0_LSB << 16) | (y0_MSB << 8) | 0x2b; data_array[2] = (y1_LSB); dsi_set_cmdq(data_array, 3, 1); data_array[0] = 0x00290508; dsi_set_cmdq(data_array, 1, 1); data_array[0] = 0x002c3909; dsi_set_cmdq(data_array, 1, 0); } #endif LCM_DRIVER kr070ia4t_dsi_vdo_lcm_drv = { .name = "KR070IA4T_DSI_VDO", .set_util_funcs = lcm_set_util_funcs, .get_params = lcm_get_params, .init = lcm_init_lcm, .suspend = lcm_suspend, .resume = lcm_resume, .init_power = lcm_init_power, .resume_power = lcm_resume_power, .suspend_power = lcm_suspend_power, #if (LCM_DSI_CMD_MODE) .update = lcm_update, #endif };