aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/video/samsung_duallcd/s3cfb_s6e39a0.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/video/samsung_duallcd/s3cfb_s6e39a0.c')
-rw-r--r--drivers/video/samsung_duallcd/s3cfb_s6e39a0.c1095
1 files changed, 1095 insertions, 0 deletions
diff --git a/drivers/video/samsung_duallcd/s3cfb_s6e39a0.c b/drivers/video/samsung_duallcd/s3cfb_s6e39a0.c
new file mode 100644
index 0000000..806baea
--- /dev/null
+++ b/drivers/video/samsung_duallcd/s3cfb_s6e39a0.c
@@ -0,0 +1,1095 @@
+/* linux/drivers/video/samsung/s3cfb_s6e8aa0.c
+ *
+ * MIPI-DSI based AMS529HA01 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"
+
+#ifdef SMART_DIMMING
+#include "smart_dimming.h"
+#endif
+
+#define DCS_PARTIAL_MODE_ON 0x12
+#define DCS_PARTIAL_AREA 0x30
+
+#define MIN_BRIGHTNESS 0
+#define MAX_BRIGHTNESS 255
+#define DEFAULT_BRIGHTNESS 150
+#define DEFAULT_GAMMA_LEVEL 10
+
+#define POWER_IS_ON(pwr) ((pwr) <= FB_BLANK_NORMAL)
+
+#define LDI_ID_REG 0xD1
+#define LDI_ID_LEN 3
+
+#ifdef SMART_DIMMING
+#define PANEL_A1_M3 0xA1
+#define PANEL_A2_M3 0xA2
+#define PANEL_A1_SM2 0x12
+
+#define LDI_MTP_LENGTH 24
+#define LDI_MTP_ADDR 0xD3
+
+#define ELVSS_OFFSET_MAX 0x00
+#define ELVSS_OFFSET_2 0x08
+#define ELVSS_OFFSET_1 0x0D
+#define ELVSS_OFFSET_MIN 0x11
+
+#define DYNAMIC_ELVSS_MIN_VALUE 0x81
+#define DYNAMIC_ELVSS_MAX_VALUE 0x9F
+
+#define ELVSS_MODE0_MIN_VOLTAGE 62
+#define ELVSS_MODE1_MIN_VOLTAGE 52
+
+struct str_elvss {
+ u8 reference;
+ u8 limit;
+};
+#endif
+
+struct lcd_info {
+ unsigned int bl;
+ unsigned int acl_enable;
+ unsigned int cur_acl;
+ unsigned int current_bl;
+
+ unsigned int ldi_enable;
+ unsigned int power;
+ struct mutex lock;
+
+ struct device *dev;
+ struct lcd_device *ld;
+ struct backlight_device *bd;
+ struct lcd_platform_data *lcd_pd;
+ struct early_suspend early_suspend;
+
+ unsigned int lcd_id;
+
+ const unsigned char **gamma_table;
+
+#ifdef SMART_DIMMING
+ unsigned int support_elvss;
+ unsigned int aid;
+ unsigned int manual_version;
+
+ struct str_smart_dim smart;
+ struct str_elvss elvss;
+ struct mutex bl_lock;
+#endif
+ unsigned int irq;
+ unsigned int connected;
+
+ struct dsim_global *dsim;
+};
+
+static int s6e8ax0_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 s6e8ax0_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 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:
+ backlightlevel = 0;
+ break;
+ case 1 ... 29:
+ backlightlevel = 0;
+ break;
+ case 30 ... 34:
+ backlightlevel = 1;
+ break;
+ case 35 ... 44:
+ backlightlevel = 2;
+ break;
+ case 45 ... 54:
+ backlightlevel = 3;
+ break;
+ case 55 ... 64:
+ backlightlevel = 4;
+ break;
+ case 65 ... 74:
+ backlightlevel = 5;
+ break;
+ case 75 ... 83:
+ backlightlevel = 6;
+ break;
+ case 84 ... 93:
+ backlightlevel = 7;
+ break;
+ case 94 ... 103:
+ backlightlevel = 8;
+ break;
+ case 104 ... 113:
+ backlightlevel = 9;
+ break;
+ case 114 ... 122:
+ backlightlevel = 10;
+ break;
+ case 123 ... 132:
+ backlightlevel = 11;
+ break;
+ case 133 ... 142:
+ backlightlevel = 12;
+ break;
+ case 143 ... 152:
+ backlightlevel = 13;
+ break;
+ case 153 ... 162:
+ backlightlevel = 14;
+ break;
+ case 163 ... 171:
+ backlightlevel = 15;
+ break;
+ case 172 ... 181:
+ backlightlevel = 16;
+ break;
+ case 182 ... 191:
+ backlightlevel = 17;
+ break;
+ case 192 ... 201:
+ backlightlevel = 18;
+ break;
+ case 202 ... 210:
+ backlightlevel = 19;
+ break;
+ case 211 ... 220:
+ backlightlevel = 20;
+ break;
+ case 221 ... 230:
+ backlightlevel = 21;
+ break;
+ case 231 ... 240:
+ backlightlevel = 22;
+ break;
+ case 241 ... 250:
+ backlightlevel = 23;
+ break;
+ case 251 ... 255:
+ backlightlevel = 24;
+ break;
+ default:
+ backlightlevel = 10;
+ break;
+ }
+ return backlightlevel;
+}
+
+static int s6e8ax0_set_acl(struct lcd_info *lcd)
+{
+ if (lcd->acl_enable) {
+ if (lcd->cur_acl == 0) {
+ if (lcd->bl == 0 || lcd->bl == 1) {
+ s6e8ax0_write(lcd, SEQ_ACL_OFF, ARRAY_SIZE(SEQ_ACL_OFF));
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ } else
+ s6e8ax0_write(lcd, SEQ_ACL_ON, ARRAY_SIZE(SEQ_ACL_ON));
+ }
+ switch (lcd->bl) {
+ case 0 ... 1: /* 30cd ~ 40cd - 0%*/
+ if (lcd->cur_acl != 0) {
+ s6e8ax0_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 2 ... 12: /* 70cd ~ 180cd -40%*/
+ if (lcd->cur_acl != 40) {
+ s6e8ax0_write(lcd, acl_cutoff_table[1], ACL_PARAM_SIZE);
+ lcd->cur_acl = 40;
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ }
+ break;
+ case 13: /* 190cd - 43% */
+ if (lcd->cur_acl != 43) {
+ s6e8ax0_write(lcd, acl_cutoff_table[2], ACL_PARAM_SIZE);
+ lcd->cur_acl = 43;
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ }
+ break;
+ case 14: /* 200cd - 45% */
+ if (lcd->cur_acl != 45) {
+ s6e8ax0_write(lcd, acl_cutoff_table[3], ACL_PARAM_SIZE);
+ lcd->cur_acl = 45;
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ }
+ break;
+ case 15: /* 210cd - 47% */
+ if (lcd->cur_acl != 47) {
+ s6e8ax0_write(lcd, acl_cutoff_table[4], ACL_PARAM_SIZE);
+ lcd->cur_acl = 47;
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ }
+ break;
+ case 16: /* 220cd - 48% */
+ if (lcd->cur_acl != 48) {
+ s6e8ax0_write(lcd, acl_cutoff_table[5], ACL_PARAM_SIZE);
+ lcd->cur_acl = 48;
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ }
+ break;
+ default:
+ if (lcd->cur_acl != 50) {
+ s6e8ax0_write(lcd, acl_cutoff_table[6], ACL_PARAM_SIZE);
+ lcd->cur_acl = 50;
+ dev_dbg(&lcd->ld->dev, "%s : cur_acl=%d\n", __func__, lcd->cur_acl);
+ }
+ break;
+ }
+ } else {
+ s6e8ax0_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);
+ }
+
+ return 0;
+}
+
+#ifndef SMART_DIMMING
+static int s6e8ax0_gamma_ctl(struct lcd_info *lcd)
+{
+ s6e8ax0_write(lcd, gamma22_table[lcd->bl], GAMMA_PARAM_SIZE);
+
+ /* Gamma Set Update */
+ s6e8ax0_write(lcd, SEQ_GAMMA_UPDATE, ARRAY_SIZE(SEQ_GAMMA_UPDATE));
+
+ lcd->current_bl = lcd->bl;
+
+ return 0;
+}
+
+static int update_brightness(struct lcd_info *lcd)
+{
+ int ret;
+
+#if defined(CONFIG_MACH_C1) || \
+ defined(CONFIG_MACH_M3) || \
+ defined(CONFIG_MACH_M0) || defined(CONFIG_MACH_SLP_PQ) || \
+ defined(CONFIG_MACH_SLP_PQ_LTE)
+#else
+ ret = s6e8ax0_gamma_ctl(lcd);
+ if (ret)
+ return -EPERM;
+
+ /*
+ ret = s6e8ax0_set_elvss(lcd);
+ if (ret)
+ return -EPERM;
+
+ ret = s6e8ax0_set_acl(lcd);
+ if (ret)
+ return -EPERM;
+ */
+#endif
+
+ return 0;
+}
+#endif
+
+#ifdef SMART_DIMMING
+static int s6e8ax0_read_mtp(struct lcd_info *lcd, u8 *mtp_data)
+{
+ int ret;
+ u8 retry_cnt = 3;
+
+read_retry:
+ ret = s6e8ax0_read(lcd, LDI_MTP_ADDR, LDI_MTP_LENGTH, mtp_data);
+ if (!ret) {
+ if (retry_cnt) {
+ printk(KERN_WARNING "[WARN:LCD] : %s : retry cnt : %d\n", __func__, retry_cnt);
+ retry_cnt--;
+ goto read_retry;
+ } else
+ printk(KERN_ERR "ERROR:MTP read failed\n");
+ }
+
+ return ret;
+}
+
+static u32 transform_gamma(u32 brightness)
+{
+ u32 gamma;
+
+ switch (brightness) {
+ case 0:
+ gamma = 30;
+ break;
+ case 1 ... 29:
+ gamma = 30;
+ break;
+ case 30 ... 34:
+ gamma = 40;
+ break;
+ case 35 ... 44:
+ gamma = 70;
+ break;
+ case 45 ... 54:
+ gamma = 80;
+ break;
+ case 55 ... 64:
+ gamma = 90;
+ break;
+ case 65 ... 74:
+ gamma = 100;
+ break;
+ case 75 ... 83:
+ gamma = 110;
+ break;
+ case 84 ... 93:
+ gamma = 120;
+ break;
+ case 94 ... 103:
+ gamma = 130;
+ break;
+ case 104 ... 113:
+ gamma = 140;
+ break;
+ case 114 ... 122:
+ gamma = 150;
+ break;
+ case 123 ... 132:
+ gamma = 160;
+ break;
+ case 133 ... 142:
+ gamma = 170;
+ break;
+ case 143 ... 152:
+ gamma = 180;
+ break;
+ case 153 ... 162:
+ gamma = 190;
+ break;
+ case 163 ... 171:
+ gamma = 200;
+ break;
+ case 172 ... 181:
+ gamma = 210;
+ break;
+ case 182 ... 191:
+ gamma = 220;
+ break;
+ case 192 ... 201:
+ gamma = 230;
+ break;
+ case 202 ... 210:
+ gamma = 240;
+ break;
+ case 211 ... 220:
+ gamma = 250;
+ break;
+ case 221 ... 230:
+ gamma = 260;
+ break;
+ case 231 ... 240:
+ gamma = 270;
+ break;
+ case 241 ... 250:
+ gamma = 280;
+ break;
+ case 251 ... 255:
+ gamma = 290;
+ break;
+ default:
+ gamma = 150;
+ break;
+ }
+ return gamma - 1;
+}
+
+static int s6e8aa0_update_brightness(struct lcd_info *lcd, u32 brightness)
+{
+ int ret = 0;
+ u8 gamma_regs[GAMMA_PARAM_SIZE] = {0,};
+ u32 gamma;
+#if 0
+ int i;
+#endif
+
+ gamma_regs[0] = 0xFA;
+ gamma_regs[1] = 0x01;
+
+ gamma = brightness;
+
+ calc_gamma_table(&lcd->smart, gamma, gamma_regs+2);
+
+ s6e8ax0_write(lcd, gamma_regs, GAMMA_PARAM_SIZE);
+
+ s6e8ax0_write(lcd, SEQ_GAMMA_UPDATE, sizeof(SEQ_GAMMA_UPDATE));
+
+#if 0
+ printk(KERN_INFO "##### print gamma reg #####\n");
+ for (i = 0; i < 26; i++)
+ printk(KERN_INFO "[%02d] : %02x\n", i, gamma_regs[i]);
+#endif
+
+ return ret;
+}
+
+static u8 get_offset_brightness(u32 candela)
+{
+ u8 offset = 0;
+
+ switch (candela) {
+ case 0 ... 100:
+ offset = ELVSS_OFFSET_MIN;
+ break;
+ case 101 ... 160:
+ offset = ELVSS_OFFSET_1;
+ break;
+ case 161 ... 200:
+ offset = ELVSS_OFFSET_2;
+ break;
+ case 201 ... 300:
+ offset = ELVSS_OFFSET_MAX;
+ break;
+ default:
+ offset = ELVSS_OFFSET_MAX;
+ break;
+ }
+ return offset;
+}
+
+static u8 get_elvss_value(struct lcd_info *lcd, u32 candela)
+{
+ u8 ref = 0;
+ u8 offset;
+
+ ref = (lcd->elvss.reference | 0x80);
+ offset = get_offset_brightness(candela);
+ ref += offset;
+
+ printk(KERN_DEBUG "%s ref =0x%x , offset = 0x%x\n", __func__, ref, offset);
+
+ if (ref < DYNAMIC_ELVSS_MIN_VALUE)
+ ref = DYNAMIC_ELVSS_MIN_VALUE;
+ else if (ref > DYNAMIC_ELVSS_MAX_VALUE)
+ ref = DYNAMIC_ELVSS_MAX_VALUE;
+
+ return ref;
+}
+
+static int s6e8aa0_update_elvss(struct lcd_info *lcd, u32 candela)
+{
+ u8 elvss_cmd[3];
+ u8 elvss;
+
+ elvss = get_elvss_value(lcd, candela);
+ if (!elvss) {
+ printk(KERN_ERR "[ERROR:LCD]:%s:get_elvss_value() failed\n", __func__);
+ return -EPERM;
+ }
+
+ elvss_cmd[0] = 0xb1;
+ elvss_cmd[1] = 0x04;
+ elvss_cmd[2] = elvss;
+
+ printk(KERN_DEBUG "elvss reg : %02x\n", elvss_cmd[2]);
+ s6e8ax0_write(lcd, elvss_cmd, sizeof(elvss_cmd));
+
+ return 0;
+}
+
+static int s6e8ax0_adb_brightness_update(struct lcd_info *lcd, u32 br, u32 force)
+{
+ u32 gamma;
+ int ret = 0;
+
+ mutex_lock(&lcd->bl_lock);
+
+ lcd->bl = get_backlight_level_from_brightness(br);
+
+ if ((force) || ((lcd->ldi_enable) && (lcd->current_bl != lcd->bl))) {
+ gamma = transform_gamma(br);
+
+ ret = s6e8aa0_update_brightness(lcd, gamma);
+
+ ret = s6e8ax0_set_acl(lcd);
+
+ if (lcd->support_elvss)
+ ret = s6e8aa0_update_elvss(lcd, gamma);
+
+ lcd->current_bl = lcd->bl;
+ dev_info(&lcd->ld->dev, "brightness=%d, gamma=%d\n", br, gamma);
+ }
+
+ mutex_unlock(&lcd->bl_lock);
+
+ return ret;
+}
+#endif
+
+static int s6e8ax0_ldi_init(struct lcd_info *lcd)
+{
+ int ret = 0;
+ s6e8ax0_write(lcd, SEQ_APPLY_LEVEL_2_KEY, \
+ ARRAY_SIZE(SEQ_APPLY_LEVEL_2_KEY));
+ s6e8ax0_write(lcd, SEQ_APPLY_LEVEL_3_KEY, \
+ ARRAY_SIZE(SEQ_APPLY_LEVEL_3_KEY));
+ s6e8ax0_write(lcd, SEQ_APPLY_LEVEL_4_KEY, \
+ ARRAY_SIZE(SEQ_APPLY_LEVEL_4_KEY));
+ s6e8ax0_write(lcd, SEQ_GAMMA_CONDITION_SET, \
+ ARRAY_SIZE(SEQ_GAMMA_CONDITION_SET));
+ s6e8ax0_write(lcd, SEQ_GAMMA_UPDATE,\
+ ARRAY_SIZE(SEQ_GAMMA_UPDATE));
+ s6e8ax0_write(lcd, SEQ_PANEL_CONDITION_SET, \
+ ARRAY_SIZE(SEQ_PANEL_CONDITION_SET));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_0, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_0));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_1, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_1));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_2, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_2));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_3, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_3));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_4, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_4));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_5, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_5));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_6, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_6));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_7, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_7));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_8, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_8));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_9, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_9));
+ s6e8ax0_write(lcd, SEQ_ETC_CONDITION_SET_2_10, \
+ ARRAY_SIZE(SEQ_ETC_CONDITION_SET_2_10));
+ s6e8ax0_write(lcd, SEQ_SLEEP_OUT, \
+ ARRAY_SIZE(SEQ_SLEEP_OUT));
+ mdelay(120);
+ s6e8ax0_write(lcd, SEQ_MEMORY_WINDOW_SET_1_0, \
+ ARRAY_SIZE(SEQ_MEMORY_WINDOW_SET_1_0));
+ s6e8ax0_write(lcd, SEQ_MEMORY_WINDOW_SET_1_1, \
+ ARRAY_SIZE(SEQ_MEMORY_WINDOW_SET_1_1));
+ mdelay(1);
+ s6e8ax0_write(lcd, SEQ_MEMORY_WINDOW_SET_2_1, \
+ ARRAY_SIZE(SEQ_MEMORY_WINDOW_SET_2_1));
+ s6e8ax0_write(lcd, SEQ_MEMORY_WINDOW_SET_2_2, \
+ ARRAY_SIZE(SEQ_MEMORY_WINDOW_SET_2_2));
+ s6e8ax0_write(lcd, SEQ_MEMORY_WINDOW_SET_2_3, \
+ ARRAY_SIZE(SEQ_MEMORY_WINDOW_SET_2_3));
+ s6e8ax0_write(lcd, SEQ_MEMORY_WINDOW_SET_2_4, \
+ ARRAY_SIZE(SEQ_MEMORY_WINDOW_SET_2_4));
+ return ret;
+}
+
+static int s6e8ax0_ldi_enable(struct lcd_info *lcd)
+{
+ int ret = 0;
+
+ s6e8ax0_write(lcd, SEQ_DISPLAY_ON, ARRAY_SIZE(SEQ_DISPLAY_ON));
+
+ return ret;
+}
+
+static int s6e8ax0_ldi_disable(struct lcd_info *lcd)
+{
+ int ret = 0;
+
+ s6e8ax0_write(lcd, SEQ_DISPLAY_OFF, ARRAY_SIZE(SEQ_DISPLAY_OFF));
+ s6e8ax0_write(lcd, SEQ_STANDBY_ON, ARRAY_SIZE(SEQ_STANDBY_ON));
+
+ return ret;
+}
+
+static int s6e8ax0_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 = s6e8ax0_ldi_init(lcd);
+
+ if (ret) {
+ dev_err(&lcd->ld->dev, "failed to initialize ldi.\n");
+ goto err;
+ }
+
+ msleep(120);
+
+ ret = s6e8ax0_ldi_enable(lcd);
+ if (ret) {
+ dev_err(&lcd->ld->dev, "failed to enable ldi.\n");
+ goto err;
+ }
+
+#ifdef SMART_DIMMING
+ lcd->ldi_enable = 1;
+
+ s6e8ax0_adb_brightness_update(lcd, lcd->bd->props.brightness, 1);
+#else
+ update_brightness(lcd);
+
+ lcd->ldi_enable = 1;
+#endif
+
+err:
+ return ret;
+}
+
+static int s6e8ax0_power_off(struct lcd_info *lcd)
+{
+ int ret = 0;
+
+ dev_info(&lcd->ld->dev, "%s\n", __func__);
+
+ lcd->ldi_enable = 0;
+
+ ret = s6e8ax0_ldi_disable(lcd);
+
+ msleep(120);
+
+ return ret;
+}
+
+static int s6e8ax0_power(struct lcd_info *lcd, int power)
+{
+ int ret = 0;
+
+ if (POWER_IS_ON(power) && !POWER_IS_ON(lcd->power))
+ ret = s6e8ax0_power_on(lcd);
+ else if (!POWER_IS_ON(power) && POWER_IS_ON(lcd->power))
+ ret = s6e8ax0_power_off(lcd);
+
+ if (!ret)
+ lcd->power = power;
+
+ return ret;
+}
+
+static int s6e8ax0_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 s6e8ax0_power(lcd, power);
+}
+
+static int s6e8ax0_get_power(struct lcd_device *ld)
+{
+ struct lcd_info *lcd = lcd_get_data(ld);
+
+ return lcd->power;
+}
+
+#ifdef SMART_DIMMING
+static int s6e8ax0_set_brightness(struct backlight_device *bd)
+{
+ int ret = 0;
+ u32 brightness = (u32)bd->props.brightness;
+
+ struct lcd_info *lcd = bl_get_data(bd);
+
+ 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;
+ }
+
+ /* dev_info(&lcd->ld->dev, "[%s] brightness=%d\n", __func__, brightness); */
+
+ ret = s6e8ax0_adb_brightness_update(lcd, brightness, 0);
+
+ return ret;
+}
+#else
+static int s6e8ax0_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;
+ }
+
+ lcd->bl = get_backlight_level_from_brightness(brightness);
+
+ if ((lcd->ldi_enable) && (lcd->current_bl != lcd->bl)) {
+ ret = update_brightness(lcd);
+ dev_info(&lcd->ld->dev, "brightness=%d, bl=%d\n",\
+ bd->props.brightness, lcd->bl);
+ if (ret < 0) {
+ dev_err(lcd->dev, "err in %s\n", __func__);
+ return -EINVAL;
+ }
+ }
+
+ return ret;
+}
+#endif
+
+static int s6e8ax0_get_brightness(struct backlight_device *bd)
+{
+ return bd->props.brightness;
+}
+
+static struct lcd_ops s6e8ax0_lcd_ops = {
+ .set_power = s6e8ax0_set_power,
+ .get_power = s6e8ax0_get_power,
+};
+
+static const struct backlight_ops s6e8ax0_backlight_ops = {
+ .get_brightness = s6e8ax0_get_brightness,
+ .update_status = s6e8ax0_set_brightness,
+};
+
+static ssize_t lcd_type_show(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ char temp[15];
+ sprintf(temp, "SMD_AMS465GS0x\n");
+ strcat(buf, temp);
+ return strlen(buf);
+}
+
+static DEVICE_ATTR(lcd_type, 0444, lcd_type_show, NULL);
+
+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;
+
+ if (lcd->acl_enable != value) {
+ dev_info(dev, "%s - %d, %d\n", __func__, \
+ lcd->acl_enable, value);
+ lcd->acl_enable = value;
+ if (lcd->ldi_enable)
+ s6e8ax0_set_acl(lcd);
+ }
+ return size;
+}
+
+static DEVICE_ATTR(power_reduce, 0664, power_reduce_show, power_reduce_store);
+
+#ifdef CONFIG_HAS_EARLYSUSPEND
+struct lcd_info *g_lcd;
+
+void s6e8ax0_early_suspend(void)
+{
+ struct lcd_info *lcd = g_lcd;
+
+ dev_info(&lcd->ld->dev, "+%s\n", __func__);
+ s6e8ax0_power(lcd, FB_BLANK_POWERDOWN);
+ dev_info(&lcd->ld->dev, "-%s\n", __func__);
+
+ return ;
+}
+
+void s6e8ax0_late_resume(void)
+{
+ struct lcd_info *lcd = g_lcd;
+
+ dev_info(&lcd->ld->dev, "+%s\n", __func__);
+ s6e8ax0_power(lcd, FB_BLANK_UNBLANK);
+ dev_info(&lcd->ld->dev, "-%s\n", __func__);
+
+ return ;
+}
+#endif
+
+static void s6e8ax0_read_id(struct lcd_info *lcd, u8 *buf)
+{
+ int ret = 0;
+ u8 retry_cnt = 3;
+
+read_retry:
+ ret = s6e8ax0_read(lcd, LDI_ID_REG, LDI_ID_LEN, buf);
+ if (!ret) {
+ if (retry_cnt) {
+ printk(KERN_WARNING "[WARN:LCD] : %s : retry cnt : %d\n",\
+ __func__, retry_cnt);
+ retry_cnt--;
+ goto read_retry;
+ } else {
+ printk(KERN_ERR "[ERROR:LCD] : %s : Read ID Failed\n", __func__);
+ lcd->connected = 0;
+ dev_info(&lcd->ld->dev, "panel is not connected well\n");
+ }
+ }
+}
+
+#ifdef SMART_DIMMING
+static void s6e8aa0_check_id(struct lcd_info *lcd, u8 *idbuf)
+{
+ int i;
+
+ for (i = 0; i < LDI_ID_LEN; i++)
+ lcd->smart.panelid[i] = idbuf[i];
+
+ lcd->aid = lcd->smart.panelid[2] & 0xe0 >> 5;
+
+ if (idbuf[0] == PANEL_A1_SM2) {
+ lcd->support_elvss = 1;
+ lcd->aid = lcd->smart.panelid[2] & 0xe0 >> 5;
+ lcd->elvss.reference = lcd->smart.panelid[2];
+
+ printk(KERN_DEBUG "Dynamic ELVSS Information\n");
+ printk(KERN_DEBUG "Refrence : %02x , manual_version = %02x, lcd->aid= %02x\n", lcd->elvss.reference, lcd->manual_version, lcd->aid);
+ } else if ((idbuf[0] == PANEL_A1_M3) || (idbuf[0] == PANEL_A2_M3)) {
+ lcd->support_elvss = 0;
+ printk(KERN_DEBUG "ID-3 is 0xff does not support dynamic elvss\n");
+ } else {
+ lcd->support_elvss = 0;
+ printk(KERN_DEBUG "No valid panel id\n");
+ }
+}
+#endif
+
+static int s6e8ax0_probe(struct device *dev)
+{
+ int ret = 0;
+ struct lcd_info *lcd;
+#ifdef SMART_DIMMING
+ u8 mtp_data[LDI_MTP_LENGTH] = {0,};
+#endif
+ u8 idbuf[LDI_ID_LEN] = {0,};
+
+ 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, &s6e8ax0_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, &s6e8ax0_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;
+
+ 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__);
+
+ dev_set_drvdata(dev, lcd);
+
+#if 0
+#ifdef CONFIG_HAS_EARLYSUSPEND
+ lcd->early_suspend.suspend = s6e8ax0_early_suspend;
+ lcd->early_suspend.resume = s6e8ax0_late_resume;
+ lcd->early_suspend.level = EARLY_SUSPEND_LEVEL_DISABLE_FB - 2;
+ register_early_suspend(&lcd->early_suspend);
+#endif
+#endif
+
+ mutex_init(&lcd->lock);
+
+ s6e8ax0_read_id(lcd, idbuf);
+
+ dev_info(&lcd->ld->dev, "ID : %x, %x, %x\n", idbuf[0], idbuf[1], idbuf[2]);
+
+ dev_info(&lcd->ld->dev, \
+ "s6e8ax0 lcd panel driver has been probed.\n");
+
+#ifdef SMART_DIMMING
+ s6e8aa0_check_id(lcd, idbuf);
+
+ init_table_info(&lcd->smart);
+
+ ret = s6e8ax0_read_mtp(lcd, mtp_data);
+ if (!ret) {
+ printk(KERN_ERR "[LCD:ERROR] : %s read mtp failed\n", __func__);
+ /*return -EPERM;*/
+ }
+
+ calc_voltage_table(&lcd->smart, mtp_data);
+
+ mutex_init(&lcd->bl_lock);
+
+ s6e8ax0_adb_brightness_update(lcd, lcd->bd->props.brightness, 1);
+#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 s6e8ax0_remove(struct device *dev)
+{
+ struct lcd_info *lcd = dev_get_drvdata(dev);
+
+ s6e8ax0_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 s6e8ax0_shutdown(struct device *dev)
+{
+ struct lcd_info *lcd = dev_get_drvdata(dev);
+
+ dev_info(&lcd->ld->dev, "%s\n", __func__);
+
+ s6e8ax0_power(lcd, FB_BLANK_POWERDOWN);
+}
+
+static struct mipi_lcd_driver s6e8ax0_mipi_driver = {
+ .name = "s6e8aa0",
+ .probe = s6e8ax0_probe,
+ .remove = __devexit_p(s6e8ax0_remove),
+ .shutdown = s6e8ax0_shutdown,
+};
+
+static int s6e8ax0_init(void)
+{
+ return s5p_dsim_register_lcd_driver(&s6e8ax0_mipi_driver);
+}
+
+static void s6e8ax0_exit(void)
+{
+ return;
+}
+
+module_init(s6e8ax0_init);
+module_exit(s6e8ax0_exit);
+
+MODULE_DESCRIPTION("MIPI-DSI S6E8AA0:AMS529HA01 (800x1280) Panel Driver");
+MODULE_LICENSE("GPL");