diff options
Diffstat (limited to 'drivers/video/samsung_duallcd/s3cfb_s6e63m0.c')
-rw-r--r-- | drivers/video/samsung_duallcd/s3cfb_s6e63m0.c | 1205 |
1 files changed, 1205 insertions, 0 deletions
diff --git a/drivers/video/samsung_duallcd/s3cfb_s6e63m0.c b/drivers/video/samsung_duallcd/s3cfb_s6e63m0.c new file mode 100644 index 0000000..01be7b1 --- /dev/null +++ b/drivers/video/samsung_duallcd/s3cfb_s6e63m0.c @@ -0,0 +1,1205 @@ +/* linux/drivers/video/samsung/s3cfb_s6e63m0.c + * + * MIPI-DSI based AMS367GEXX AMOLED lcd panel driver. + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + * +*/ + +#include <linux/module.h> +#include <linux/kernel.h> +#include <linux/errno.h> +#include <linux/mutex.h> +#include <linux/wait.h> +#include <linux/ctype.h> +#include <linux/io.h> +#include <linux/delay.h> +#include <linux/irq.h> +#include <linux/interrupt.h> +#include <linux/gpio.h> +#include <linux/workqueue.h> +#include <linux/backlight.h> +#include <linux/lcd.h> +#include <plat/gpio-cfg.h> +#include <plat/regs-dsim.h> +#include <mach/dsim.h> +#include <mach/mipi_ddi.h> +#ifdef CONFIG_HAS_EARLYSUSPEND +#include <linux/earlysuspend.h> +#endif + +#include "s5p-dsim.h" +#include "s3cfb.h" +#include "s6e63m0_gamma_l.h" + +#include "s6e63m0_gamma_grande.h" +#define SMART_DIMMING + +#ifdef SMART_DIMMING +#include "smart_mtp_s6e63m0.h" +#endif + +#define POWER_IS_ON(pwr) ((pwr) <= FB_BLANK_NORMAL) + +#define MIN_BRIGHTNESS 0 +#define MAX_BRIGHTNESS 255 +#define MAX_GAMMA 300 +#define DEFAULT_BRIGHTNESS 160 +#define DEFAULT_GAMMA_LEVEL GAMMA_160CD + +#define LDI_ID_REG 0xDA +#define LDI_ID_LEN 3 + +#ifdef SMART_DIMMING +#define PANEL_A1_SM2 0xA1 + +#define LDI_MTP_LENGTH 24 +#define LDI_MTP_ADDR 0xD3 + +#define DYNAMIC_ELVSS_MIN_VALUE 0x81 +#define DYNAMIC_ELVSS_MAX_VALUE 0x9F + +#define ELVSS_MODE0_MIN_VOLTAGE 62 +#define ELVSS_MODE1_MIN_VOLTAGE 52 + +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) +#define NR_S6E63M0_PANEL (2) +#else /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ +#define NR_S6E63M0_PANEL (1) +#endif /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ + +struct str_elvss { + u8 reference; + u8 limit; +}; +#endif + +struct lcd_info *g_lcd; + +struct lcd_info { + unsigned int bl; + unsigned int auto_brightness; + unsigned int acl_enable; + unsigned int cur_acl; + unsigned int current_bl; + unsigned int current_elvss; + + unsigned int ldi_enable; + unsigned int power; + struct mutex lock; + struct mutex bl_lock; + + struct device *dev; + struct lcd_device *ld; + struct backlight_device *bd; + struct lcd_platform_data *lcd_pd; + struct early_suspend early_suspend; + + unsigned char id[LDI_ID_LEN]; + + unsigned char **gamma_table[NR_S6E63M0_PANEL]; + unsigned char **elvss_table[NR_S6E63M0_PANEL]; + +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) + unsigned int detected[NR_S6E63M0_PANEL]; + unsigned int panel_select; +#endif /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ + +#ifdef SMART_DIMMING + unsigned int support_elvss; + + struct SMART_DIM smart; + struct str_elvss elvss; +#endif + unsigned int irq; + unsigned int connected; + + struct dsim_global *dsim; +}; + +static const unsigned int candela_table[GAMMA_MAX] = { + 30, 40, 50, 60, 70, 80, 90, 100, 110, 120, + 130, 140, 150, 160, 170, 180, 190, 200, 210, 220, + 230, 240, 250, MAX_GAMMA +}; + +/* Added for DUAL LCD work*/ +static int s6e63m0_read_mtp(struct lcd_info *lcd); +#ifdef SMART_DIMMING_DEBUG_LCD +static int print_mtp_value(struct lcd_info *lcd); +#endif +static int s6e63m0_read_id(struct lcd_info *lcd, u8 *buf); +static int s6e63m0_dual_panel_detect(struct lcd_info *lcd); +/* Added for DUAL LCD work*/ + +#if defined(GPIO_OLED_DET) +struct delayed_work hs_clk_re_try; +unsigned int count_dsim; + +static void hs_clk_re_try_work(struct work_struct *work) +{ + int read_oled_det; + + read_oled_det = gpio_get_value(GPIO_OLED_DET); + printk(KERN_INFO "%s, %d, %d\n", __func__, + count_dsim, read_oled_det); + if (read_oled_det == 1) { + if (count_dsim < 10) { + schedule_delayed_work(&hs_clk_re_try, HZ/8); + count_dsim++; + set_dsim_hs_clk_toggle_count(15); + } else + set_dsim_hs_clk_toggle_count(0); + } else + set_dsim_hs_clk_toggle_count(0); +} + +static irqreturn_t oled_det_int(int irq, void *dev_id) +{ + pr_info("[DSIM] %s\n", __func__); + schedule_delayed_work(&hs_clk_re_try, HZ/16); + count_dsim = 0; + return IRQ_HANDLED; +} +#endif + +static int s6e63m0_write(struct lcd_info *lcd, + const unsigned char *seq, int len) +{ + int size; + const unsigned char *wbuf; + + if (!lcd->connected) + return 0; + mutex_lock(&lcd->lock); + + size = len; + wbuf = seq; + if (size == 1) + lcd->dsim->ops->cmd_write(lcd->dsim, DCS_WR_NO_PARA, wbuf[0], 0); + else if (size == 2) + lcd->dsim->ops->cmd_write(lcd->dsim, DCS_WR_1_PARA, wbuf[0], wbuf[1]); + else + lcd->dsim->ops->cmd_write(lcd->dsim, DCS_LONG_WR, (unsigned int)wbuf, size); + mutex_unlock(&lcd->lock); + + return 0; +} + +static int _s6e63m0_read(struct lcd_info *lcd, const u8 addr, + u16 count, u8 *buf) +{ + int ret = 0; + + if (!lcd->connected) + return ret; + + mutex_lock(&lcd->lock); + if (lcd->dsim->ops->cmd_read) + ret = lcd->dsim->ops->cmd_read(lcd->dsim, addr, count, buf); + mutex_unlock(&lcd->lock); + + return ret; +} + +static int s6e63m0_read(struct lcd_info *lcd, const u8 addr, + u16 count, u8 *buf, u8 retry_cnt) +{ + int ret = 0; +read_retry: + ret = _s6e63m0_read(lcd, addr, count, buf); + if (!ret) { + if (retry_cnt) { + pr_info("[WARN:LCD] %s : retry cnt : %d\n", + __func__, retry_cnt); + retry_cnt--; + goto read_retry; + } else + pr_info("[ERROR:LCD] %s : 0x%02x read failed\n", + __func__, addr); + } + return ret; +} + +static int get_backlight_level_from_brightness(int brightness) +{ + int backlightlevel; + + /* brightness setting from platform is from 0 to 255 + * But in this driver, brightness is + only supported from 0 to 24 */ + + switch (brightness) { + case 0 ... 29: + backlightlevel = GAMMA_30CD; + break; + case 30 ... 254: + backlightlevel = (brightness - candela_table[0]) / 10; + break; + case 255: + backlightlevel = ARRAY_SIZE(candela_table) - 1; + break; + default: + backlightlevel = DEFAULT_GAMMA_LEVEL; + break; + } + return backlightlevel; +} + +static int s6e63m0_gamma_ctl(struct lcd_info *lcd) +{ +#ifdef SMART_DIMMING_DEBUG_LCD + int j; +#endif + s6e63m0_write(lcd, lcd->gamma_table[lcd->panel_select][lcd->bl], + GAMMA_PARAM_SIZE); + + s6e63m0_write(lcd, SEQ_GAMMA_UPDATE, + ARRAY_SIZE(SEQ_GAMMA_UPDATE)); +#ifdef SMART_DIMMING_DEBUG_LCD + pr_info("panel_select = %d bl = %d\n", lcd->panel_select, lcd->bl); + for (j = 0; j < GAMMA_PARAM_SIZE; j++) + pr_info("0x%02x, ", + lcd->gamma_table[lcd->panel_select][lcd->bl][j]); +#endif + return 0; +} + +static int s6e63m0_set_acl(struct lcd_info *lcd) +{ + if (lcd->acl_enable) { + if (lcd->cur_acl == 0) { + if (lcd->bl == 0 || lcd->bl == 1) { + s6e63m0_write(lcd, SEQ_ACL_OFF, ARRAY_SIZE(SEQ_ACL_OFF)); + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d, acl_off\n", + __func__, lcd->cur_acl); + } else { + s6e63m0_write(lcd, SEQ_ACL_ON, ARRAY_SIZE(SEQ_ACL_ON)); + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d, acl_on\n", + __func__, lcd->cur_acl); + } + } + switch (lcd->bl) { + case GAMMA_30CD ... GAMMA_40CD: + s6e63m0_write(lcd, SEQ_ACL_OFF, ARRAY_SIZE(SEQ_ACL_OFF)); + lcd->cur_acl = 0; + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", + __func__, lcd->cur_acl); + break; + case GAMMA_50CD ... GAMMA_160CD: + s6e63m0_write(lcd, ACL_CUTOFF_TABLE[ACL_STATUS_40P], ACL_PARAM_SIZE); + lcd->cur_acl = 40; + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", + __func__, lcd->cur_acl); + break; + case GAMMA_170CD ... GAMMA_220CD: + s6e63m0_write(lcd, ACL_CUTOFF_TABLE[ACL_STATUS_48P], ACL_PARAM_SIZE); + lcd->cur_acl = 48; + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", + __func__, lcd->cur_acl); + break; + case GAMMA_230CD ... GAMMA_300CD: + s6e63m0_write(lcd, ACL_CUTOFF_TABLE[ACL_STATUS_50P], ACL_PARAM_SIZE); + lcd->cur_acl = 50; + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", + __func__, lcd->cur_acl); + break; + default: + lcd->cur_acl = 0; + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl); + break; + } + } else { + s6e63m0_write(lcd, SEQ_ACL_OFF, ARRAY_SIZE(SEQ_ACL_OFF)); + lcd->cur_acl = 0; + dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d, acl_off\n", + __func__, lcd->cur_acl); + } + return 0; +} + +#ifdef SMART_DIMMING +static int s6e63m0_set_elvss(struct lcd_info *lcd) +{ + int ret = 0, elvss_level = 0; + u32 candela = candela_table[lcd->bl]; + + switch (candela) { + case 0 ... 100: + elvss_level = ELVSS_MIN; + break; + case 101 ... 160: + elvss_level = ELVSS_1; + break; + case 161 ... 200: + elvss_level = ELVSS_2; + break; + case 201 ... 300: + elvss_level = ELVSS_MAX; + break; + default: + break; + } + if (lcd->current_elvss != lcd->elvss_table[lcd->panel_select][elvss_level][1]) { + ret = s6e63m0_write(lcd, lcd->elvss_table[lcd->panel_select][elvss_level], + ELVSS_PARAM_SIZE); + lcd->current_elvss = lcd->elvss_table[lcd->panel_select][elvss_level][1]; + } + dev_dbg(&lcd->ld->dev, "elvss = %x\n", + lcd->elvss_table[lcd->panel_select][elvss_level][1]); + if (ret) { + ret = -EPERM; + goto elvss_err; + } + +elvss_err: + return ret; +} +#endif +static u8 get_elvss_value(struct lcd_info *lcd, u8 elvss_level) +{ + u8 alph[ELVSS_STATUS_MAX] = {0xb, 0x8, 0x6, 0x0}; + u8 data = 0; + u8 value = 0; + + value = lcd->id[2]; + data = value + alph[elvss_level]; + if (data > 0x29) + data = 0x29; + + return data; +} + +static int alloc_elvss_table(struct lcd_info *lcd) +{ + int i, k, ret = 0; + for (k = 0; k < NR_S6E63M0_PANEL; k++) { + lcd->elvss_table[k] + = kzalloc(ELVSS_STATUS_MAX * sizeof(u8 *), GFP_KERNEL); + + if (IS_ERR_OR_NULL(lcd->elvss_table[k])) { + pr_err("failed to allocate elvss table\n"); + ret = -ENOMEM; + goto err_alloc_elvss_table; + } + + for (i = 0; i < ELVSS_STATUS_MAX; i++) { + lcd->elvss_table[k][i] + = kzalloc(ELVSS_PARAM_SIZE * sizeof(u8), + GFP_KERNEL); + if (IS_ERR_OR_NULL(lcd->elvss_table[k][i])) { + pr_err("failed to allocate elvss\n"); + ret = -ENOMEM; + goto err_alloc_elvss; + } + } + } + return 0; + +err_alloc_elvss: + while (i > 0) { + kfree(lcd->elvss_table[i-1]); + i--; + } + kfree(lcd->elvss_table); +err_alloc_elvss_table: + return ret; +} + +static int init_elvss_table(struct lcd_info *lcd) +{ + int i, j; + for (i = 0; i < ELVSS_STATUS_MAX; i++) { + lcd->elvss_table[lcd->panel_select][i][0] = 0xB2; + for (j = 1; j < ELVSS_PARAM_SIZE; j++) + lcd->elvss_table[lcd->panel_select][i][j] + = get_elvss_value(lcd, i); + } +#ifdef SMART_DIMMING_DEBUG_LCD + pr_info("ELVSS_TABLE [%2d]\n", lcd->panel_select); + for (i = 0; i < ELVSS_STATUS_MAX; i++) { + for (j = 0; j < ELVSS_PARAM_SIZE; j++) + pr_info("0x%02x, ", + lcd->elvss_table[lcd->panel_select][i][j]); + pr_info("\n"); + } +#endif + return 0; +} + +static int alloc_gamma_table(struct lcd_info *lcd) +{ + int i, k, ret = 0; + for (k = 0; k < NR_S6E63M0_PANEL; k++) { + lcd->gamma_table[k] = kzalloc(GAMMA_MAX * sizeof(u8 *), + GFP_KERNEL); + if (IS_ERR_OR_NULL(lcd->gamma_table[k])) { + pr_err("failed to allocate gamma table\n"); + ret = -ENOMEM; + goto err_alloc_gamma_table; + } + + for (i = 0; i < GAMMA_MAX; i++) { + lcd->gamma_table[k][i] + = kzalloc(GAMMA_PARAM_SIZE * sizeof(u8), + GFP_KERNEL); + if (IS_ERR_OR_NULL(lcd->gamma_table[k][i])) { + pr_err("failed to allocate gamma\n"); + ret = -ENOMEM; + goto err_alloc_gamma; + } + } + } + return 0; + +err_alloc_gamma: + while (i > 0) { + kfree(lcd->gamma_table[i-1]); + i--; + } + kfree(lcd->gamma_table); +err_alloc_gamma_table: + return ret; +} + +static int init_gamma_table(struct lcd_info *lcd) +{ + int i, j; + char gen_gamma[GAMMA_PARAM_SIZE] = {0,}; + + for (i = 0; i < GAMMA_MAX; i++) { + lcd->gamma_table[lcd->panel_select][i][0] = 0xFA; + lcd->gamma_table[lcd->panel_select][i][1] = 0x02; + + lcd->smart.brightness_level = candela_table[i]; + generate_gamma(&(lcd->smart), gen_gamma, GAMMA_PARAM_SIZE); + for (j = 2; j < GAMMA_PARAM_SIZE; j++) + lcd->gamma_table[lcd->panel_select][i][j] = gen_gamma[j - 2]; + } +#ifdef SMART_DIMMING_DEBUG_LCD + pr_info("GAMMA_TABLE[%2d]\n", lcd->panel_select); + for (i = 0; i < GAMMA_MAX; i++) { + pr_info("%3d : ", i); + for (j = 0; j < GAMMA_PARAM_SIZE; j++) + pr_info("0x%02x, ", + lcd->gamma_table[lcd->panel_select][i][j]); + pr_info("\n"); + } +#endif + return 0; +} + +static int update_brightness(struct lcd_info *lcd, u8 force) +{ + int ret; +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) + int sel; +#endif /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ + u32 brightness; + + mutex_lock(&lcd->bl_lock); + + brightness = lcd->bd->props.brightness; + +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) + sel = s5p_dsim_get_panel_sel_value(); + lcd->panel_select = sel; +#endif /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ + + if (unlikely(!lcd->auto_brightness && brightness > 250)) + brightness = 250; + + lcd->bl = get_backlight_level_from_brightness(brightness); + + if ((force) || ((lcd->ldi_enable) && + (lcd->current_bl != lcd->bl))) { + + ret = s6e63m0_set_elvss(lcd); + ret = s6e63m0_set_acl(lcd); + ret = s6e63m0_gamma_ctl(lcd); + + lcd->current_bl = lcd->bl; + dev_info(&lcd->ld->dev, + "panel = %d brightness=%d, bl=%d, candela=%d\n", + lcd->panel_select, brightness, + lcd->bl, candela_table[lcd->bl]); + } + + mutex_unlock(&lcd->bl_lock); + + return 0; +} + +static int s6e63m0_mtp_read_cmds(struct lcd_info *lcd) +{ + s6e63m0_write(lcd, SEQ_PREPARE_MTP_READ1, + ARRAY_SIZE(SEQ_PREPARE_MTP_READ1)); + s6e63m0_write(lcd, SEQ_PREPARE_MTP_READ2, + ARRAY_SIZE(SEQ_PREPARE_MTP_READ2)); + s6e63m0_write(lcd, SEQ_SLEEP_OUT, + ARRAY_SIZE(SEQ_SLEEP_OUT)); + mdelay(120); + s6e63m0_write(lcd, SEQ_CONTENTION_ERROR_REMOVE, + ARRAY_SIZE(SEQ_CONTENTION_ERROR_REMOVE)); + return 0; +} + +static int s6e63m0_ldi_init(struct lcd_info *lcd) +{ + int ret = 0; + + mdelay(25); /* 25ms */ + s6e63m0_write(lcd, SEQ_SW_RESET, + ARRAY_SIZE(SEQ_SW_RESET)); /* SW Reset */ + mdelay(5); /* Wait 5ms more */ + s6e63m0_write(lcd, SEQ_APPLY_LEVEL2_KEY_ENABLE, + ARRAY_SIZE(SEQ_APPLY_LEVEL2_KEY_ENABLE)); + s6e63m0_write(lcd, SEQ_APPLY_MTP_KEY_ENABLE, + ARRAY_SIZE(SEQ_APPLY_MTP_KEY_ENABLE)); + s6e63m0_write(lcd, SEQ_SLEEP_OUT, + ARRAY_SIZE(SEQ_SLEEP_OUT)); + mdelay(10); /* 10ms */ + s6e63m0_write(lcd, SEQ_PANEL_CONDITION_SET, + ARRAY_SIZE(SEQ_PANEL_CONDITION_SET)); + s6e63m0_write(lcd, SEQ_DISPLAY_CONDITION_SET1, + ARRAY_SIZE(SEQ_DISPLAY_CONDITION_SET1)); + s6e63m0_write(lcd, SEQ_DISPLAY_CONDITION_SET2, + ARRAY_SIZE(SEQ_DISPLAY_CONDITION_SET2)); + s6e63m0_write(lcd, lcd->gamma_table[lcd->panel_select][lcd->bl], + GAMMA_PARAM_SIZE); + s6e63m0_write(lcd, SEQ_GAMMA_UPDATE, + ARRAY_SIZE(SEQ_GAMMA_UPDATE)); + s6e63m0_write(lcd, SEQ_ETC_SOURCE_CONTROL, + ARRAY_SIZE(SEQ_ETC_SOURCE_CONTROL)); + s6e63m0_write(lcd, SEQ_ETC_CONTROL_B3h, + ARRAY_SIZE(SEQ_ETC_CONTROL_B3h)); + s6e63m0_write(lcd, SEQ_ELVSS_SET, ARRAY_SIZE(SEQ_ELVSS_SET)); + s6e63m0_write(lcd, SEQ_ELVSS_ON, ARRAY_SIZE(SEQ_ELVSS_ON)); + return ret; +} + +static int s6e63m0_ldi_enable(struct lcd_info *lcd) +{ + int ret = 0; + s6e63m0_write(lcd, SEQ_DISPLAY_ON, ARRAY_SIZE(SEQ_DISPLAY_ON)); + return ret; +} + +static int s6e63m0_ldi_disable(struct lcd_info *lcd) +{ + int ret = 0; + s6e63m0_write(lcd, SEQ_DISPLAY_OFF, + ARRAY_SIZE(SEQ_DISPLAY_OFF)); + s6e63m0_write(lcd, SEQ_STANDBY_ON, + ARRAY_SIZE(SEQ_STANDBY_ON)); + return ret; +} + +static int s6e63m0_power_on(struct lcd_info *lcd) +{ + int ret = 0; + struct lcd_platform_data *pd = NULL; + pd = lcd->lcd_pd; + + dev_info(&lcd->ld->dev, "%s\n", __func__); + + ret = s6e63m0_ldi_init(lcd); + if (ret) { + dev_err(&lcd->ld->dev, "failed to initialize ldi.\n"); + goto err; + } + + msleep(120); + + ret = s6e63m0_ldi_enable(lcd); + if (ret) { + dev_err(&lcd->ld->dev, "failed to enable ldi.\n"); + goto err; + } + lcd->ldi_enable = 1; + +#if !defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) + update_brightness(lcd, 1); +#endif +err: + return ret; +} + +static int s6e63m0_power_off(struct lcd_info *lcd) +{ + int ret = 0; + + dev_info(&lcd->ld->dev, "%s\n", __func__); + lcd->ldi_enable = 0; + ret = s6e63m0_ldi_disable(lcd); + msleep(135); + + return ret; +} + +static int s6e63m0_power(struct lcd_info *lcd, int power) +{ + int ret = 0; + + if (POWER_IS_ON(power) && !POWER_IS_ON(lcd->power)) + ret = s6e63m0_power_on(lcd); + else if (!POWER_IS_ON(power) && POWER_IS_ON(lcd->power)) + ret = s6e63m0_power_off(lcd); + if (!ret) + lcd->power = power; + + return ret; +} + +static int s6e63m0_set_power(struct lcd_device *ld, int power) +{ + struct lcd_info *lcd = lcd_get_data(ld); + + if (power != FB_BLANK_UNBLANK && power != FB_BLANK_POWERDOWN && + power != FB_BLANK_NORMAL) { + dev_err(&lcd->ld->dev, "power value should be 0, 1 or 4.\n"); + return -EINVAL; + } + + return s6e63m0_power(lcd, power); +} + +static int s6e63m0_get_power(struct lcd_device *ld) +{ + struct lcd_info *lcd = lcd_get_data(ld); + + return lcd->power; +} + +int s6e63m0_sleep_in(void) +{ + struct lcd_info *lcd = g_lcd; + int ret; + + ret = s6e63m0_ldi_disable(lcd); + return ret; +} + +static int s6e63m0_set_brightness(struct backlight_device *bd) +{ + int ret = 0; + int brightness = bd->props.brightness; + struct lcd_info *lcd = bl_get_data(bd); + + /* dev_info(&lcd->ld->dev, "%s: brightness=%d\n", + __func__, brightness); */ + + if (brightness < MIN_BRIGHTNESS || + brightness > bd->props.max_brightness) { + dev_err(&bd->dev, "lcd brightness should be %d to %d. now %d\n", + MIN_BRIGHTNESS, MAX_BRIGHTNESS, brightness); + return -EINVAL; + } + + if (lcd->ldi_enable) { + ret = update_brightness(lcd, 0); + if (ret < 0) { + dev_err(lcd->dev, "err in %s\n", __func__); + return -EINVAL; + } + } + + return ret; +} + +static int s6e63m0_get_brightness(struct backlight_device *bd) +{ + struct lcd_info *lcd = bl_get_data(bd); + + return candela_table[lcd->bl]; +} + +static struct lcd_ops s6e63m0_lcd_ops = { + .set_power = s6e63m0_set_power, + .get_power = s6e63m0_get_power, +}; + +static const struct backlight_ops s6e63m0_backlight_ops = { + .get_brightness = s6e63m0_get_brightness, + .update_status = s6e63m0_set_brightness, +}; + +static ssize_t power_reduce_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + char temp[3]; + + sprintf(temp, "%d\n", lcd->acl_enable); + strcpy(buf, temp); + + return strlen(buf); +} + +static ssize_t power_reduce_store(struct device *dev, + struct device_attribute *attr, const char *buf, size_t size) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + int value; + int rc; + + rc = strict_strtoul(buf, (unsigned int)0, + (unsigned long *)&value); + if (rc < 0) + return rc; + else { + if (lcd->acl_enable != value) { + dev_info(dev, "%s - %d, %d\n", + __func__, lcd->acl_enable, value); + mutex_lock(&lcd->bl_lock); + lcd->acl_enable = value; + if (lcd->ldi_enable) + s6e63m0_set_acl(lcd); + mutex_unlock(&lcd->bl_lock); + } + } + return size; +} + +static DEVICE_ATTR(power_reduce, 0664, power_reduce_show, + power_reduce_store); + +static ssize_t lcd_type_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + char temp[20]; + sprintf(temp, "SMD_AMS397GEXX\n"); + strcat(buf, temp); + return strlen(buf); +} + +static DEVICE_ATTR(lcd_type, 0444, lcd_type_show, NULL); + +static ssize_t gamma_table_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + int i, j, k; + + for (k = 0; k < NR_S6E63M0_PANEL ; k++) { + for (i = 0; i < GAMMA_MAX; i++) { + for (j = 0; j < GAMMA_PARAM_SIZE; j++) + pr_info("0x%02x, ", lcd->gamma_table[k][i][j]); + pr_info("\n"); + } + + for (i = 0; i < ELVSS_STATUS_MAX; i++) { + for (j = 0; j < ELVSS_PARAM_SIZE; j++) + pr_info("0x%02x, ", lcd->elvss_table[k][i][j]); + pr_info("\n"); + } + } + + return strlen(buf); +} +static DEVICE_ATTR(gamma_table, 0444, gamma_table_show, NULL); + +static ssize_t auto_brightness_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + char temp[3]; + + sprintf(temp, "%d\n", lcd->auto_brightness); + strcpy(buf, temp); + + return strlen(buf); +} + +static ssize_t auto_brightness_store(struct device *dev, + struct device_attribute *attr, const char *buf, size_t size) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + int value; + int rc; + + rc = strict_strtoul(buf, (unsigned int)0, (unsigned long *)&value); + if (rc < 0) + return rc; + else { + if (lcd->auto_brightness != value) { + dev_info(dev, "%s - %d, %d\n", + __func__, + lcd->auto_brightness, + value); + mutex_lock(&lcd->bl_lock); + lcd->auto_brightness = value; + mutex_unlock(&lcd->bl_lock); + if (lcd->ldi_enable) + update_brightness(lcd, 0); + } + } + return size; +} + +static DEVICE_ATTR(auto_brightness, 0644, auto_brightness_show, + auto_brightness_store); + +#ifdef CONFIG_HAS_EARLYSUSPEND +struct lcd_info *g_lcd; + +void s6e63m0_early_suspend(void) +{ + struct lcd_info *lcd = g_lcd; + set_dsim_lcd_enabled(0); + + dev_info(&lcd->ld->dev, "+%s\n", __func__); +#if defined(GPIO_OLED_DET) + disable_irq(lcd->irq); + gpio_request(GPIO_OLED_DET, "OLED_DET"); + s3c_gpio_cfgpin(GPIO_OLED_DET, S3C_GPIO_OUTPUT); + s3c_gpio_setpull(GPIO_OLED_DET, S3C_GPIO_PULL_NONE); + gpio_direction_output(GPIO_OLED_DET, GPIO_LEVEL_LOW); + gpio_free(GPIO_OLED_DET); +#endif + s6e63m0_power(lcd, FB_BLANK_POWERDOWN); + dev_info(&lcd->ld->dev, "-%s\n", __func__); + + return ; +} + +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) +static int s6e63m0_dual_panel_detect(struct lcd_info *lcd) +{ + int ret = 0; + + /* Calculating Gamma table of outer LCD */ + lcd->panel_select = s5p_dsim_get_panel_sel_value(); + msleep(25); + if (lcd->gamma_table[lcd->panel_select][0][0] == 0x00) { + pr_info("GAMMA TABLE not found !! in panel %d\n", + lcd->panel_select); + ret = s6e63m0_read_id(lcd, lcd->id); + if (lcd->id[0] != 0xFE) { + lcd->detected[lcd->panel_select] = 0; + goto dual_det_err; + } else + lcd->detected[lcd->panel_select] = 1; + dev_info(&lcd->ld->dev, "ID: %x, %x, %x\n", lcd->id[0], + lcd->id[1], lcd->id[2]); + dev_info(&lcd->ld->dev, "s6e63m0 driver has been detected.\n"); + ret = s6e63m0_read_mtp(lcd); + if (!ret) { + pr_info("[LCD:ERROR] : %s read mtp failed\n", + __func__); + } +#ifdef SMART_DIMMING_DEBUG_LCD + ret += print_mtp_value(lcd); +#endif + ret += Smart_dimming_init(&(lcd->smart)); + if (lcd->support_elvss) + ret = init_elvss_table(lcd); + ret += init_gamma_table(lcd); + if (ret) { + lcd->gamma_table[lcd->panel_select] + = (unsigned char **)s6e63m0_gamma22_table; + lcd->elvss_table[lcd->panel_select] + = (unsigned char **)ELVSS_TABLE; + } + } + return 0; +dual_det_err: + pr_info("Outer LCD detectoin failed\n"); + ret = s5p_dsim_toggle_lcd(); + return 0; +} +#endif /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ + +void s6e63m0_late_resume(void) +{ + struct lcd_info *lcd = g_lcd; + + dev_info(&lcd->ld->dev, "+%s\n", __func__); + s6e63m0_power(lcd, FB_BLANK_UNBLANK); +#if defined(GPIO_OLED_DET) + s3c_gpio_cfgpin(GPIO_OLED_DET, S3C_GPIO_SFN(0xf)); + s3c_gpio_setpull(GPIO_OLED_DET, S3C_GPIO_PULL_NONE); + enable_irq(lcd->irq); +#endif + dev_info(&lcd->ld->dev, "-%s\n", __func__); + + set_dsim_lcd_enabled(1); +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) + if (lcd->detected[lcd->panel_select] == 0) + s6e63m0_dual_panel_detect(lcd); + update_brightness(lcd, 1); +#endif + return ; +} +#endif + +static int s6e63m0_id_read_cmds(struct lcd_info *lcd) +{ + s6e63m0_write(lcd, SEQ_PREPARE_ID_READ, + ARRAY_SIZE(SEQ_PREPARE_ID_READ)); + return 0; +} + +static int s6e63m0_read_id(struct lcd_info *lcd, u8 *buf) +{ + int ret = 0; + + s6e63m0_id_read_cmds(lcd); + /* ID1: Default value of panel */ + ret += s6e63m0_read(lcd, 0xDA, 1, buf, 3); + /* ID2: Way to Cell working */ + ret += s6e63m0_read(lcd, 0xDB, 1, buf + 1, 3); + /* Olny ID2 == C1 support */ + if (*(buf + 1) == 0xC1) + ret += s6e63m0_read(lcd, 0xDC, 1, buf + 2, 3); + else + *(buf + 2) = 0x00; + if (!ret) { + lcd->connected = 0; + dev_info(&lcd->ld->dev, "panel is not connected well\n"); + } + return ret; +} + +#ifdef SMART_DIMMING +static int s6e63m0_read_mtp(struct lcd_info *lcd) +{ + int ret; + s6e63m0_mtp_read_cmds(lcd); + ret = s6e63m0_read(lcd, LDI_MTP_ADDR, + LDI_MTP_LENGTH, (u8 *)(&(lcd->smart.MTP)), 0); + return ret; +} + +static void s6e63m0_check_id(struct lcd_info *lcd, u8 *idbuf) +{ + if (idbuf[0] == PANEL_A1_SM2) + lcd->support_elvss = 0; + else { + lcd->support_elvss = 1; + lcd->elvss.reference = idbuf[2] & (BIT(0) | BIT(1) | + BIT(2) | BIT(3) | BIT(4)); + pr_info("Dynamic ELVSS Information, 0x%x\n", + lcd->elvss.reference); + } +} + +#ifdef SMART_DIMMING_DEBUG_LCD +static int print_mtp_value(struct lcd_info *lcd) +{ + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_1); + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_19); + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_43); + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_87); + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_171); + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_255_MSB); + pr_info("0x%02x, ", lcd->smart.MTP.R_OFFSET.OFFSET_255_LSB); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_1); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_19); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_43); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_87); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_171); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_255_MSB); + pr_info("0x%02x, ", lcd->smart.MTP.G_OFFSET.OFFSET_255_LSB); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_1); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_19); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_43); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_87); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_171); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_255_MSB); + pr_info("0x%02x, ", lcd->smart.MTP.B_OFFSET.OFFSET_255_LSB); + return 0; +} +#endif + +#endif + +static int s6e63m0_probe(struct device *dev) +{ + int ret = 0; + struct lcd_info *lcd; + + lcd = kzalloc(sizeof(struct lcd_info), GFP_KERNEL); + if (!lcd) { + pr_err("failed to allocate for lcd\n"); + ret = -ENOMEM; + goto err_alloc; + } + + g_lcd = lcd; + + lcd->ld = lcd_device_register("panel", dev, lcd, &s6e63m0_lcd_ops); + if (IS_ERR(lcd->ld)) { + pr_err("failed to register lcd device\n"); + ret = PTR_ERR(lcd->ld); + goto out_free_lcd; + } + + lcd->bd = backlight_device_register("panel", dev, lcd, + &s6e63m0_backlight_ops, NULL); + if (IS_ERR(lcd->bd)) { + pr_err("failed to register backlight device\n"); + ret = PTR_ERR(lcd->bd); + goto out_free_backlight; + } + + lcd->dev = dev; + lcd->dsim = (struct dsim_global *)dev_get_drvdata(dev->parent); + lcd->bd->props.max_brightness = MAX_BRIGHTNESS; + lcd->bd->props.brightness = DEFAULT_BRIGHTNESS; + lcd->bl = DEFAULT_GAMMA_LEVEL; + lcd->current_bl = lcd->bl; + + lcd->acl_enable = 0; + lcd->cur_acl = 0; + + lcd->power = FB_BLANK_UNBLANK; + lcd->ldi_enable = 1; + lcd->connected = 1; + lcd->auto_brightness = 0; + + ret = device_create_file(&lcd->ld->dev, &dev_attr_power_reduce); + if (ret < 0) + dev_err(&lcd->ld->dev, "failed to add sysfs entries, %d\n", + __LINE__); + + ret = device_create_file(&lcd->ld->dev, &dev_attr_lcd_type); + if (ret < 0) + dev_err(&lcd->ld->dev, "failed to add sysfs entries, %d\n", + __LINE__); + + ret = device_create_file(&lcd->ld->dev, &dev_attr_gamma_table); + if (ret < 0) + dev_err(&lcd->ld->dev, "failed to add sysfs entries, %d\n", + __LINE__); + + ret = device_create_file(&lcd->bd->dev, &dev_attr_auto_brightness); + if (ret < 0) + dev_err(&lcd->ld->dev, "failed to add sysfs entries, %d\n", + __LINE__); + + dev_set_drvdata(dev, lcd); + + mutex_init(&lcd->lock); + mutex_init(&lcd->bl_lock); + /* Start to reading DDI for Smart dimming */ + lcd->panel_select = s5p_dsim_get_panel_sel_value(); + lcd->detected[0] = 0; +#if defined(CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD) + lcd->detected[1] = 0; +#endif /* CONFIG_S5P_DSIM_SWITCHABLE_DUAL_LCD */ + ret += s6e63m0_read_id(lcd, lcd->id); + if (lcd->id[0] != 0xFE) { + lcd->detected[lcd->panel_select] = 0; + goto err_alloc; + + } else + lcd->detected[lcd->panel_select] = 1; + + dev_info(&lcd->ld->dev, "ID: %x, %x, %x\n", lcd->id[0], + lcd->id[1], lcd->id[2]); + dev_info(&lcd->ld->dev, "s6e63m0 driver has been probed.\n"); + +#ifdef SMART_DIMMING + s6e63m0_check_id(lcd, lcd->id); + ret = s6e63m0_read_mtp(lcd); + if (!ret) { + pr_info("[LCD:ERROR] : %s read mtp failed\n", + __func__); + } +#ifdef SMART_DIMMING_DEBUG_LCD + ret += print_mtp_value(lcd); +#endif + ret += Smart_dimming_init(&(lcd->smart)); + if (lcd->support_elvss) { + ret = alloc_elvss_table(lcd); + ret = init_elvss_table(lcd); + } + + if (ret) { + lcd->elvss_table[lcd->panel_select] + = (unsigned char **)ELVSS_TABLE; + } + ret += alloc_gamma_table(lcd); + ret += init_gamma_table(lcd); + if (ret) { + lcd->gamma_table[lcd->panel_select] + = (unsigned char **)s6e63m0_gamma22_table; + } + /* End reading DDI for Smart dimming */ + update_brightness(lcd, 1); +#endif + +#if defined(GPIO_OLED_DET) + if (lcd->connected) { + INIT_DELAYED_WORK(&hs_clk_re_try, hs_clk_re_try_work); + + lcd->irq = gpio_to_irq(GPIO_OLED_DET); + + s3c_gpio_cfgpin(GPIO_OLED_DET, S3C_GPIO_SFN(0xf)); + s3c_gpio_setpull(GPIO_OLED_DET, S3C_GPIO_PULL_NONE); + if (request_irq(lcd->irq, oled_det_int, + IRQF_TRIGGER_RISING, + "esd_detection", 0)) + pr_err("failed to reqeust irq. %d\n", lcd->irq); + } +#endif + return 0; + +out_free_backlight: + lcd_device_unregister(lcd->ld); + kfree(lcd); + return ret; + +out_free_lcd: + kfree(lcd); + return ret; + +err_alloc: + return ret; +} + +static int __devexit s6e63m0_remove(struct device *dev) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + + s6e63m0_power(lcd, FB_BLANK_POWERDOWN); + lcd_device_unregister(lcd->ld); + backlight_device_unregister(lcd->bd); + kfree(lcd); + + return 0; +} + +/* Power down all displays on reboot, poweroff or halt. */ +static void s6e63m0_shutdown(struct device *dev) +{ + struct lcd_info *lcd = dev_get_drvdata(dev); + dev_info(&lcd->ld->dev, "%s\n", __func__); + s6e63m0_power(lcd, FB_BLANK_POWERDOWN); +} + +static struct mipi_lcd_driver s6e63m0_mipi_driver = { + .name = "s6e63m0", + .probe = s6e63m0_probe, + .remove = __devexit_p(s6e63m0_remove), + .shutdown = s6e63m0_shutdown, +}; + +static int s6e63m0_init(void) +{ + return s5p_dsim_register_lcd_driver(&s6e63m0_mipi_driver); +} + +static void s6e63m0_exit(void) +{ + return; +} + +module_init(s6e63m0_init); +module_exit(s6e63m0_exit); + +MODULE_DESCRIPTION("MIPI-DSI S6E63M0:AMS367GEXX (480X800) Panel Driver"); +MODULE_LICENSE("GPL"); |