1
0
mirror of git://projects.qi-hardware.com/openwrt-xburst.git synced 2025-01-25 11:41:05 +02:00

836 lines
21 KiB
C
Raw Normal View History

/* Linux kernel driver for the tpo JBT6K74-AS LCM ASIC
*
* Copyright (C) 2006-2007 by Openmoko, Inc.
* Author: Harald Welte <laforge@openmoko.org>,
* Stefan Schmidt <stefan@openmoko.org>
* Copyright (C) 2008 by Harald Welte <laforge@openmoko.org>
* Copyright (C) 2009 by Lars-Peter Clausen <lars@metafoo.de>
* All rights reserved.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2 of
* the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*/
#include <linux/kernel.h>
#include <linux/types.h>
#include <linux/module.h>
#include <linux/device.h>
#include <linux/platform_device.h>
#include <linux/delay.h>
#include <linux/workqueue.h>
#include <linux/jbt6k74.h>
#include <linux/fb.h>
#include <linux/lcd.h>
#include <linux/time.h>
enum jbt_register {
JBT_REG_SLEEP_IN = 0x10,
JBT_REG_SLEEP_OUT = 0x11,
JBT_REG_DISPLAY_OFF = 0x28,
JBT_REG_DISPLAY_ON = 0x29,
JBT_REG_RGB_FORMAT = 0x3a,
JBT_REG_QUAD_RATE = 0x3b,
JBT_REG_POWER_ON_OFF = 0xb0,
JBT_REG_BOOSTER_OP = 0xb1,
JBT_REG_BOOSTER_MODE = 0xb2,
JBT_REG_BOOSTER_FREQ = 0xb3,
JBT_REG_OPAMP_SYSCLK = 0xb4,
JBT_REG_VSC_VOLTAGE = 0xb5,
JBT_REG_VCOM_VOLTAGE = 0xb6,
JBT_REG_EXT_DISPL = 0xb7,
JBT_REG_OUTPUT_CONTROL = 0xb8,
JBT_REG_DCCLK_DCEV = 0xb9,
JBT_REG_DISPLAY_MODE1 = 0xba,
JBT_REG_DISPLAY_MODE2 = 0xbb,
JBT_REG_DISPLAY_MODE = 0xbc,
JBT_REG_ASW_SLEW = 0xbd,
JBT_REG_DUMMY_DISPLAY = 0xbe,
JBT_REG_DRIVE_SYSTEM = 0xbf,
JBT_REG_SLEEP_OUT_FR_A = 0xc0,
JBT_REG_SLEEP_OUT_FR_B = 0xc1,
JBT_REG_SLEEP_OUT_FR_C = 0xc2,
JBT_REG_SLEEP_IN_LCCNT_D = 0xc3,
JBT_REG_SLEEP_IN_LCCNT_E = 0xc4,
JBT_REG_SLEEP_IN_LCCNT_F = 0xc5,
JBT_REG_SLEEP_IN_LCCNT_G = 0xc6,
JBT_REG_GAMMA1_FINE_1 = 0xc7,
JBT_REG_GAMMA1_FINE_2 = 0xc8,
JBT_REG_GAMMA1_INCLINATION = 0xc9,
JBT_REG_GAMMA1_BLUE_OFFSET = 0xca,
/* VGA */
JBT_REG_BLANK_CONTROL = 0xcf,
JBT_REG_BLANK_TH_TV = 0xd0,
JBT_REG_CKV_ON_OFF = 0xd1,
JBT_REG_CKV_1_2 = 0xd2,
JBT_REG_OEV_TIMING = 0xd3,
JBT_REG_ASW_TIMING_1 = 0xd4,
JBT_REG_ASW_TIMING_2 = 0xd5,
/* QVGA */
JBT_REG_BLANK_CONTROL_QVGA = 0xd6,
JBT_REG_BLANK_TH_TV_QVGA = 0xd7,
JBT_REG_CKV_ON_OFF_QVGA = 0xd8,
JBT_REG_CKV_1_2_QVGA = 0xd9,
JBT_REG_OEV_TIMING_QVGA = 0xde,
JBT_REG_ASW_TIMING_1_QVGA = 0xdf,
JBT_REG_ASW_TIMING_2_QVGA = 0xe0,
JBT_REG_HCLOCK_VGA = 0xec,
JBT_REG_HCLOCK_QVGA = 0xed,
};
enum jbt_resolution {
JBT_RESOLUTION_VGA,
JBT_RESOLUTION_QVGA,
};
enum jbt_power_mode {
JBT_POWER_MODE_DEEP_STANDBY,
JBT_POWER_MODE_SLEEP,
JBT_POWER_MODE_NORMAL,
};
static const char *jbt_power_mode_names[] = {
[JBT_POWER_MODE_DEEP_STANDBY] = "deep-standby",
[JBT_POWER_MODE_SLEEP] = "sleep",
[JBT_POWER_MODE_NORMAL] = "normal",
};
static const char *jbt_resolution_names[] = {
[JBT_RESOLUTION_VGA] = "vga",
[JBT_RESOLUTION_QVGA] = "qvga",
};
struct jbt_info {
struct mutex lock; /* protects this structure */
enum jbt_resolution resolution;
enum jbt_power_mode power_mode;
enum jbt_power_mode suspend_mode;
int suspended;
struct spi_device *spi_dev;
struct lcd_device *lcd_dev;
unsigned long last_sleep;
struct delayed_work blank_work;
int blank_mode;
u16 tx_buf[4];
u16 reg_cache[0xEE];
};
#define JBT_COMMAND 0x000
#define JBT_DATA 0x100
static int jbt_reg_write_nodata(struct jbt_info *jbt, u8 reg)
{
int rc;
jbt->tx_buf[0] = JBT_COMMAND | reg;
rc = spi_write(jbt->spi_dev, (u8 *)jbt->tx_buf,
1*sizeof(u16));
if (rc == 0)
jbt->reg_cache[reg] = 0;
else
dev_err(&jbt->spi_dev->dev, "jbt_reg_write_nodata spi_write ret %d\n",
rc);
return rc;
}
static int jbt_reg_write(struct jbt_info *jbt, u8 reg, u8 data)
{
int rc;
jbt->tx_buf[0] = JBT_COMMAND | reg;
jbt->tx_buf[1] = JBT_DATA | data;
rc = spi_write(jbt->spi_dev, (u8 *)jbt->tx_buf,
2*sizeof(u16));
if (rc == 0)
jbt->reg_cache[reg] = data;
else
dev_err(&jbt->spi_dev->dev, "jbt_reg_write spi_write ret %d\n", rc);
return rc;
}
static int jbt_reg_write16(struct jbt_info *jbt, u8 reg, u16 data)
{
int rc;
jbt->tx_buf[0] = JBT_COMMAND | reg;
jbt->tx_buf[1] = JBT_DATA | (data >> 8);
jbt->tx_buf[2] = JBT_DATA | (data & 0xff);
rc = spi_write(jbt->spi_dev, (u8 *)jbt->tx_buf,
3*sizeof(u16));
if (rc == 0)
jbt->reg_cache[reg] = data;
else
dev_err(&jbt->spi_dev->dev, "jbt_reg_write16 spi_write ret %d\n", rc);
return rc;
}
static int jbt_init_regs(struct jbt_info *jbt)
{
int rc;
dev_dbg(&jbt->spi_dev->dev, "entering %cVGA mode\n",
jbt->resolution == JBT_RESOLUTION_QVGA ? 'Q' : ' ');
rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE1, 0x01);
rc |= jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE2, 0x00);
rc |= jbt_reg_write(jbt, JBT_REG_RGB_FORMAT, 0x60);
rc |= jbt_reg_write(jbt, JBT_REG_DRIVE_SYSTEM, 0x10);
rc |= jbt_reg_write(jbt, JBT_REG_BOOSTER_OP, 0x56);
rc |= jbt_reg_write(jbt, JBT_REG_BOOSTER_MODE, 0x33);
rc |= jbt_reg_write(jbt, JBT_REG_BOOSTER_FREQ, 0x11);
rc |= jbt_reg_write(jbt, JBT_REG_OPAMP_SYSCLK, 0x02);
rc |= jbt_reg_write(jbt, JBT_REG_VSC_VOLTAGE, 0x2b);
rc |= jbt_reg_write(jbt, JBT_REG_VCOM_VOLTAGE, 0x40);
rc |= jbt_reg_write(jbt, JBT_REG_EXT_DISPL, 0x03);
rc |= jbt_reg_write(jbt, JBT_REG_DCCLK_DCEV, 0x04);
/*
* default of 0x02 in JBT_REG_ASW_SLEW responsible for 72Hz requirement
* to avoid red / blue flicker
*/
rc |= jbt_reg_write(jbt, JBT_REG_ASW_SLEW, 0x04 | (1 << 5));
rc |= jbt_reg_write(jbt, JBT_REG_DUMMY_DISPLAY, 0x00);
rc |= jbt_reg_write(jbt, JBT_REG_SLEEP_OUT_FR_A, 0x11);
rc |= jbt_reg_write(jbt, JBT_REG_SLEEP_OUT_FR_B, 0x11);
rc |= jbt_reg_write(jbt, JBT_REG_SLEEP_OUT_FR_C, 0x11);
rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_D, 0x2040);
rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_E, 0x60c0);
rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_F, 0x1020);
rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_G, 0x60c0);
rc |= jbt_reg_write16(jbt, JBT_REG_GAMMA1_FINE_1, 0x5533);
rc |= jbt_reg_write(jbt, JBT_REG_GAMMA1_FINE_2, 0x00);
rc |= jbt_reg_write(jbt, JBT_REG_GAMMA1_INCLINATION, 0x00);
rc |= jbt_reg_write(jbt, JBT_REG_GAMMA1_BLUE_OFFSET, 0x00);
if (jbt->resolution != JBT_RESOLUTION_QVGA) {
rc |= jbt_reg_write16(jbt, JBT_REG_HCLOCK_VGA, 0x1f0);
rc |= jbt_reg_write(jbt, JBT_REG_BLANK_CONTROL, 0x02);
rc |= jbt_reg_write16(jbt, JBT_REG_BLANK_TH_TV, 0x0804);
rc |= jbt_reg_write(jbt, JBT_REG_CKV_ON_OFF, 0x01);
rc |= jbt_reg_write16(jbt, JBT_REG_CKV_1_2, 0x0000);
rc |= jbt_reg_write16(jbt, JBT_REG_OEV_TIMING, 0x0d0e);
rc |= jbt_reg_write16(jbt, JBT_REG_ASW_TIMING_1, 0x11a4);
rc |= jbt_reg_write(jbt, JBT_REG_ASW_TIMING_2, 0x0e);
} else {
rc |= jbt_reg_write16(jbt, JBT_REG_HCLOCK_QVGA, 0x00ff);
rc |= jbt_reg_write(jbt, JBT_REG_BLANK_CONTROL_QVGA, 0x02);
rc |= jbt_reg_write16(jbt, JBT_REG_BLANK_TH_TV_QVGA, 0x0804);
rc |= jbt_reg_write(jbt, JBT_REG_CKV_ON_OFF_QVGA, 0x01);
rc |= jbt_reg_write16(jbt, JBT_REG_CKV_1_2_QVGA, 0x0008);
rc |= jbt_reg_write16(jbt, JBT_REG_OEV_TIMING_QVGA, 0x050a);
rc |= jbt_reg_write16(jbt, JBT_REG_ASW_TIMING_1_QVGA, 0x0a19);
rc |= jbt_reg_write(jbt, JBT_REG_ASW_TIMING_2_QVGA, 0x0a);
}
return rc ? -EIO : 0;
}
static int standby_to_sleep(struct jbt_info *jbt)
{
int rc;
/* three times command zero */
rc = jbt_reg_write_nodata(jbt, 0x00);
mdelay(1);
rc |= jbt_reg_write_nodata(jbt, 0x00);
mdelay(1);
rc |= jbt_reg_write_nodata(jbt, 0x00);
mdelay(1);
/* deep standby out */
rc |= jbt_reg_write(jbt, JBT_REG_POWER_ON_OFF, 0x11);
mdelay(1);
rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE, 0x28);
/* (re)initialize register set */
rc |= jbt_init_regs(jbt);
return rc ? -EIO : 0;
}
static int sleep_to_normal(struct jbt_info *jbt)
{
int rc;
/* Make sure we are 120 ms after SLEEP_OUT */
if (time_before(jiffies, jbt->last_sleep))
mdelay(jiffies_to_msecs(jbt->last_sleep - jiffies));
if (jbt->resolution == JBT_RESOLUTION_VGA) {
/* RGB I/F on, RAM wirte off, QVGA through, SIGCON enable */
rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE, 0x80);
/* Quad mode off */
rc |= jbt_reg_write(jbt, JBT_REG_QUAD_RATE, 0x00);
} else {
/* RGB I/F on, RAM wirte off, QVGA through, SIGCON enable */
rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE, 0x81);
/* Quad mode on */
rc |= jbt_reg_write(jbt, JBT_REG_QUAD_RATE, 0x22);
}
/* AVDD on, XVDD on */
rc |= jbt_reg_write(jbt, JBT_REG_POWER_ON_OFF, 0x16);
/* Output control */
rc |= jbt_reg_write16(jbt, JBT_REG_OUTPUT_CONTROL, 0xfff9);
/* Turn on display */
rc |= jbt_reg_write_nodata(jbt, JBT_REG_DISPLAY_ON);
/* Sleep mode off */
rc |= jbt_reg_write_nodata(jbt, JBT_REG_SLEEP_OUT);
jbt->last_sleep = jiffies + msecs_to_jiffies(120);
/* Allow the booster and display controller to restart stably */
mdelay(5);
return rc ? -EIO : 0;
}
static int normal_to_sleep(struct jbt_info *jbt)
{
int rc;
/* Make sure we are 120 ms after SLEEP_OUT */
while (time_before(jiffies, jbt->last_sleep))
cpu_relax();
rc = jbt_reg_write_nodata(jbt, JBT_REG_DISPLAY_OFF);
rc |= jbt_reg_write16(jbt, JBT_REG_OUTPUT_CONTROL, 0x8000 | 1 << 3);
rc |= jbt_reg_write_nodata(jbt, JBT_REG_SLEEP_IN);
jbt->last_sleep = jiffies + msecs_to_jiffies(120);
/* Allow the internal circuits to stop automatically */
mdelay(5);
return rc ? -EIO : 0;
}
static int sleep_to_standby(struct jbt_info *jbt)
{
return jbt_reg_write(jbt, JBT_REG_POWER_ON_OFF, 0x00);
}
int jbt6k74_enter_power_mode(struct jbt_info *jbt, enum jbt_power_mode new_mode)
{
struct jbt6k74_platform_data *pdata = jbt->spi_dev->dev.platform_data;
int rc = -EINVAL;
dev_dbg(&jbt->spi_dev->dev, "entering (old_state=%s, new_state=%s)\n",
jbt_power_mode_names[jbt->power_mode],
jbt_power_mode_names[new_mode]);
mutex_lock(&jbt->lock);
if (jbt->suspended) {
switch (new_mode) {
case JBT_POWER_MODE_DEEP_STANDBY:
case JBT_POWER_MODE_SLEEP:
case JBT_POWER_MODE_NORMAL:
rc = 0;
jbt->suspend_mode = new_mode;
break;
default:
break;
}
} else if (new_mode == JBT_POWER_MODE_NORMAL &&
pdata->enable_pixel_clock) {
pdata->enable_pixel_clock(&jbt->spi_dev->dev, 1);
}
switch (jbt->power_mode) {
case JBT_POWER_MODE_DEEP_STANDBY:
switch (new_mode) {
case JBT_POWER_MODE_DEEP_STANDBY:
rc = 0;
break;
case JBT_POWER_MODE_SLEEP:
rc = standby_to_sleep(jbt);
break;
case JBT_POWER_MODE_NORMAL:
/* first transition into sleep */
rc = standby_to_sleep(jbt);
/* then transition into normal */
rc |= sleep_to_normal(jbt);
break;
}
break;
case JBT_POWER_MODE_SLEEP:
switch (new_mode) {
case JBT_POWER_MODE_SLEEP:
rc = 0;
break;
case JBT_POWER_MODE_DEEP_STANDBY:
rc = sleep_to_standby(jbt);
break;
case JBT_POWER_MODE_NORMAL:
rc = sleep_to_normal(jbt);
break;
}
break;
case JBT_POWER_MODE_NORMAL:
switch (new_mode) {
case JBT_POWER_MODE_NORMAL:
rc = 0;
break;
case JBT_POWER_MODE_DEEP_STANDBY:
/* first transition into sleep */
rc = normal_to_sleep(jbt);
/* then transition into deep standby */
rc |= sleep_to_standby(jbt);
break;
case JBT_POWER_MODE_SLEEP:
rc = normal_to_sleep(jbt);
break;
}
}
if (rc == 0) {
jbt->power_mode = new_mode;
if (new_mode != JBT_POWER_MODE_NORMAL &&
pdata->enable_pixel_clock)
pdata->enable_pixel_clock(&jbt->spi_dev->dev, 0);
} else {
dev_err(&jbt->spi_dev->dev, "Failed enter state '%s')\n",
jbt_power_mode_names[new_mode]);
}
mutex_unlock(&jbt->lock);
return rc;
}
EXPORT_SYMBOL_GPL(jbt6k74_enter_power_mode);
int jbt6k74_set_resolution(struct jbt_info *jbt, enum jbt_resolution new_resolution) {
int rc = 0;
enum jbt_resolution old_resolution;
if (new_resolution != JBT_RESOLUTION_VGA &&
new_resolution != JBT_RESOLUTION_QVGA)
return -EINVAL;
mutex_lock(&jbt->lock);
if (jbt->resolution == new_resolution)
goto out_unlock;
old_resolution = jbt->resolution;
jbt->resolution = new_resolution;
if (jbt->power_mode == JBT_POWER_MODE_NORMAL) {
/* first transition into sleep */
rc = normal_to_sleep(jbt);
/* second transition into deep standby */
/* rc |= sleep_to_standby(jbt);*/
/* third transition into sleep */
/* rc |= standby_to_sleep(jbt);*/
/* fourth transition into normal */
rc |= sleep_to_normal(jbt);
if (rc) {
jbt->resolution = old_resolution;
dev_err(&jbt->spi_dev->dev, "Failed to set resolution '%s')\n",
jbt_resolution_names[new_resolution]);
}
}
out_unlock:
mutex_unlock(&jbt->lock);
return rc;
}
EXPORT_SYMBOL_GPL(jbt6k74_set_resolution);
static ssize_t resolution_read(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct jbt_info *jbt = dev_get_drvdata(dev);
if (jbt->resolution >= ARRAY_SIZE(jbt_resolution_names))
return -EIO;
return sprintf(buf, "%s\n", jbt_resolution_names[jbt->resolution]);
}
static ssize_t resolution_write(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
{
struct jbt_info *jbt = dev_get_drvdata(dev);
int i, rc;
for (i = 0; i < ARRAY_SIZE(jbt_resolution_names); i++) {
if (!strncmp(buf, jbt_resolution_names[i],
strlen(jbt_resolution_names[i]))) {
rc = jbt6k74_set_resolution(jbt, i);
if (rc)
return rc;
return count;
}
}
return -EINVAL;
}
static DEVICE_ATTR(resolution, 0644, resolution_read, resolution_write);
static int reg_by_string(const char *name)
{
if (!strcmp(name, "gamma_fine1"))
return JBT_REG_GAMMA1_FINE_1;
else if (!strcmp(name, "gamma_fine2"))
return JBT_REG_GAMMA1_FINE_2;
else if (!strcmp(name, "gamma_inclination"))
return JBT_REG_GAMMA1_INCLINATION;
else
return JBT_REG_GAMMA1_BLUE_OFFSET;
}
static ssize_t gamma_read(struct device *dev, struct device_attribute *attr,
char *buf)
{
struct jbt_info *jbt = dev_get_drvdata(dev);
int reg = reg_by_string(attr->attr.name);
u16 val;
mutex_lock(&jbt->lock);
val = jbt->reg_cache[reg];
mutex_unlock(&jbt->lock);
return sprintf(buf, "0x%04x\n", val);
}
static ssize_t gamma_write(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
{
struct jbt_info *jbt = dev_get_drvdata(dev);
int reg = reg_by_string(attr->attr.name);
unsigned long val = simple_strtoul(buf, NULL, 10);
dev_info(dev, "writing gama %lu\n", val & 0xff);
mutex_lock(&jbt->lock);
jbt_reg_write(jbt, reg, val & 0xff);
mutex_unlock(&jbt->lock);
return count;
}
static ssize_t reset_write(struct device *dev, struct device_attribute *attr,
const char *buf, size_t count)
{
int rc;
struct jbt_info *jbt = dev_get_drvdata(dev);
struct jbt6k74_platform_data *pdata = jbt->spi_dev->dev.platform_data;
dev_info(dev, "reset\n");
mutex_lock(&jbt->lock);
/* hard reset the jbt6k74 */
(pdata->reset)(0, 0);
mdelay(1);
(pdata->reset)(0, 1);
mdelay(120);
rc = jbt_reg_write_nodata(jbt, 0x01);
if (rc < 0)
dev_err(&jbt->spi_dev->dev, "cannot soft reset\n");
mdelay(120);
mutex_unlock(&jbt->lock);
jbt6k74_enter_power_mode(jbt, jbt->power_mode);
return count;
}
static DEVICE_ATTR(gamma_fine1, 0644, gamma_read, gamma_write);
static DEVICE_ATTR(gamma_fine2, 0644, gamma_read, gamma_write);
static DEVICE_ATTR(gamma_inclination, 0644, gamma_read, gamma_write);
static DEVICE_ATTR(gamma_blue_offset, 0644, gamma_read, gamma_write);
static DEVICE_ATTR(reset, 0600, NULL, reset_write);
static struct attribute *jbt_sysfs_entries[] = {
&dev_attr_resolution.attr,
&dev_attr_gamma_fine1.attr,
&dev_attr_gamma_fine2.attr,
&dev_attr_gamma_inclination.attr,
&dev_attr_gamma_blue_offset.attr,
&dev_attr_reset.attr,
NULL,
};
static struct attribute_group jbt_attr_group = {
.name = NULL,
.attrs = jbt_sysfs_entries,
};
/* FIXME: This in an ugly hack to delay display blanking.
When the jbt is in sleep mode it displays an all white screen and thus one
will a see a short flash.
By delaying the blanking we will give the backlight a chance to turn off and
thus avoid getting the flash */
static void jbt_blank_worker(struct work_struct *work) {
struct jbt_info *jbt = container_of(work, struct jbt_info,
blank_work.work);
switch (jbt->blank_mode) {
case FB_BLANK_NORMAL:
jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_SLEEP);
break;
case FB_BLANK_POWERDOWN:
jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_DEEP_STANDBY);
break;
default:
break;
}
}
static int jbt6k74_set_mode(struct lcd_device *ld, struct fb_videomode *m) {
int rc = -EINVAL;
struct jbt_info *jbt = dev_get_drvdata(&ld->dev);
if (m->xres == 240 && m->yres == 320) {
rc = jbt6k74_set_resolution(jbt, JBT_RESOLUTION_QVGA);
} else if (m->xres == 480 && m->yres == 640) {
rc = jbt6k74_set_resolution(jbt, JBT_RESOLUTION_VGA);
} else {
dev_err(&jbt->spi_dev->dev, "Unknown resolution. Entering sleep mode.\n");
jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_SLEEP);
}
return rc;
}
static int jbt6k74_set_power(struct lcd_device *ld, int power) {
int rc = -EINVAL;
struct jbt_info *jbt = dev_get_drvdata(&ld->dev);
jbt->blank_mode = power;
cancel_rearming_delayed_work(&jbt->blank_work);
switch (power) {
case FB_BLANK_UNBLANK:
dev_dbg(&jbt->spi_dev->dev, "unblank\n");
rc = jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_NORMAL);
break;
case FB_BLANK_NORMAL:
dev_dbg(&jbt->spi_dev->dev, "blank\n");
rc = schedule_delayed_work(&jbt->blank_work, HZ);
break;
case FB_BLANK_POWERDOWN:
dev_dbg(&jbt->spi_dev->dev, "powerdown\n");
rc = schedule_delayed_work(&jbt->blank_work, HZ);
break;
default:
break;
}
return rc;
}
static int jbt6k74_get_power(struct lcd_device *ld) {
struct jbt_info *jbt = dev_get_drvdata(&ld->dev);
switch (jbt->power_mode) {
case JBT_POWER_MODE_NORMAL:
return FB_BLANK_UNBLANK;
case JBT_POWER_MODE_SLEEP:
return FB_BLANK_NORMAL;
default:
return JBT_POWER_MODE_DEEP_STANDBY;
}
}
struct lcd_ops jbt6k74_lcd_ops = {
.set_power = jbt6k74_set_power,
.get_power = jbt6k74_get_power,
.set_mode = jbt6k74_set_mode,
};
/* linux device model infrastructure */
static int __devinit jbt_probe(struct spi_device *spi)
{
int rc;
struct jbt_info *jbt;
struct jbt6k74_platform_data *pdata = spi->dev.platform_data;
/* the controller doesn't have a MISO pin; we can't do detection */
spi->mode = SPI_CPOL | SPI_CPHA;
spi->bits_per_word = 9;
rc = spi_setup(spi);
if (rc < 0) {
dev_err(&spi->dev,
"error during spi_setup of jbt6k74 driver\n");
return rc;
}
jbt = kzalloc(sizeof(*jbt), GFP_KERNEL);
if (!jbt)
return -ENOMEM;
jbt->spi_dev = spi;
jbt->lcd_dev = lcd_device_register("jbt6k74-lcd", &spi->dev,
jbt, &jbt6k74_lcd_ops);
if (IS_ERR(jbt->lcd_dev)) {
rc = PTR_ERR(jbt->lcd_dev);
goto err_free_drvdata;
}
INIT_DELAYED_WORK(&jbt->blank_work, jbt_blank_worker);
jbt->resolution = JBT_RESOLUTION_VGA;
jbt->power_mode = JBT_POWER_MODE_DEEP_STANDBY;
jbt->last_sleep = jiffies + msecs_to_jiffies(120);
mutex_init(&jbt->lock);
dev_set_drvdata(&spi->dev, jbt);
rc = jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_NORMAL);
if (rc < 0) {
dev_err(&spi->dev, "cannot enter NORMAL state\n");
goto err_unregister_lcd;
}
rc = sysfs_create_group(&spi->dev.kobj, &jbt_attr_group);
if (rc < 0) {
dev_err(&spi->dev, "cannot create sysfs group\n");
goto err_standby;
}
if (pdata->probe_completed)
(pdata->probe_completed)(&spi->dev);
return 0;
err_standby:
jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_DEEP_STANDBY);
err_unregister_lcd:
lcd_device_unregister(jbt->lcd_dev);
err_free_drvdata:
dev_set_drvdata(&spi->dev, NULL);
kfree(jbt);
return rc;
}
static int __devexit jbt_remove(struct spi_device *spi)
{
struct jbt_info *jbt = dev_get_drvdata(&spi->dev);
/* We don't want to switch off the display in case the user
* accidentially unloads the module (whose use count normally is 0) */
jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_NORMAL);
sysfs_remove_group(&spi->dev.kobj, &jbt_attr_group);
dev_set_drvdata(&spi->dev, NULL);
lcd_device_unregister(jbt->lcd_dev);
kfree(jbt);
return 0;
}
#ifdef CONFIG_PM
static int jbt_suspend(struct spi_device *spi, pm_message_t state)
{
struct jbt_info *jbt = dev_get_drvdata(&spi->dev);
jbt->suspend_mode = jbt->power_mode;
jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_DEEP_STANDBY);
jbt->suspended = 1;
dev_info(&spi->dev, "suspended\n");
return 0;
}
int jbt6k74_resume(struct spi_device *spi)
{
struct jbt_info *jbt = dev_get_drvdata(&spi->dev);
jbt->suspended = 0;
jbt6k74_enter_power_mode(jbt, jbt->suspend_mode);
dev_info(&spi->dev, "resumed\n");
return 0;
}
EXPORT_SYMBOL_GPL(jbt6k74_resume);
#else
#define jbt_suspend NULL
#define jbt6k74_resume NULL
#endif
static struct spi_driver jbt6k74_driver = {
.driver = {
.name = "jbt6k74",
.owner = THIS_MODULE,
},
.probe = jbt_probe,
.remove = __devexit_p(jbt_remove),
.suspend = jbt_suspend,
.resume = jbt6k74_resume,
};
static int __init jbt_init(void)
{
return spi_register_driver(&jbt6k74_driver);
}
static void __exit jbt_exit(void)
{
spi_unregister_driver(&jbt6k74_driver);
}
MODULE_DESCRIPTION("SPI driver for tpo JBT6K74-AS LCM control interface");
MODULE_AUTHOR("Harald Welte <laforge@openmoko.org>");
MODULE_LICENSE("GPL");
module_init(jbt_init);
module_exit(jbt_exit);