Root/target/linux/s3c24xx/files-2.6.30/drivers/video/display/jbt6k74.c

1/* Linux kernel driver for the tpo JBT6K74-AS LCM ASIC
2 *
3 * Copyright (C) 2006-2007 by Openmoko, Inc.
4 * Author: Harald Welte <laforge@openmoko.org>,
5 * Stefan Schmidt <stefan@openmoko.org>
6 * Copyright (C) 2008 by Harald Welte <laforge@openmoko.org>
7 * Copyright (C) 2009 by Lars-Peter Clausen <lars@metafoo.de>
8 * All rights reserved.
9 *
10 * This program is free software; you can redistribute it and/or
11 * modify it under the terms of the GNU General Public License as
12 * published by the Free Software Foundation; either version 2 of
13 * the License, or (at your option) any later version.
14 *
15 * This program is distributed in the hope that it will be useful,
16 * but WITHOUT ANY WARRANTY; without even the implied warranty of
17 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 * GNU General Public License for more details.
19 *
20 * You should have received a copy of the GNU General Public License
21 * along with this program; if not, write to the Free Software
22 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
23 * MA 02111-1307 USA
24 *
25 */
26
27#include <linux/kernel.h>
28#include <linux/types.h>
29#include <linux/module.h>
30#include <linux/device.h>
31#include <linux/platform_device.h>
32#include <linux/delay.h>
33#include <linux/workqueue.h>
34#include <linux/jbt6k74.h>
35#include <linux/fb.h>
36#include <linux/lcd.h>
37#include <linux/time.h>
38
39enum jbt_register {
40    JBT_REG_SLEEP_IN = 0x10,
41    JBT_REG_SLEEP_OUT = 0x11,
42
43    JBT_REG_DISPLAY_OFF = 0x28,
44    JBT_REG_DISPLAY_ON = 0x29,
45
46    JBT_REG_RGB_FORMAT = 0x3a,
47    JBT_REG_QUAD_RATE = 0x3b,
48
49    JBT_REG_POWER_ON_OFF = 0xb0,
50    JBT_REG_BOOSTER_OP = 0xb1,
51    JBT_REG_BOOSTER_MODE = 0xb2,
52    JBT_REG_BOOSTER_FREQ = 0xb3,
53    JBT_REG_OPAMP_SYSCLK = 0xb4,
54    JBT_REG_VSC_VOLTAGE = 0xb5,
55    JBT_REG_VCOM_VOLTAGE = 0xb6,
56    JBT_REG_EXT_DISPL = 0xb7,
57    JBT_REG_OUTPUT_CONTROL = 0xb8,
58    JBT_REG_DCCLK_DCEV = 0xb9,
59    JBT_REG_DISPLAY_MODE1 = 0xba,
60    JBT_REG_DISPLAY_MODE2 = 0xbb,
61    JBT_REG_DISPLAY_MODE = 0xbc,
62    JBT_REG_ASW_SLEW = 0xbd,
63    JBT_REG_DUMMY_DISPLAY = 0xbe,
64    JBT_REG_DRIVE_SYSTEM = 0xbf,
65
66    JBT_REG_SLEEP_OUT_FR_A = 0xc0,
67    JBT_REG_SLEEP_OUT_FR_B = 0xc1,
68    JBT_REG_SLEEP_OUT_FR_C = 0xc2,
69    JBT_REG_SLEEP_IN_LCCNT_D = 0xc3,
70    JBT_REG_SLEEP_IN_LCCNT_E = 0xc4,
71    JBT_REG_SLEEP_IN_LCCNT_F = 0xc5,
72    JBT_REG_SLEEP_IN_LCCNT_G = 0xc6,
73
74    JBT_REG_GAMMA1_FINE_1 = 0xc7,
75    JBT_REG_GAMMA1_FINE_2 = 0xc8,
76    JBT_REG_GAMMA1_INCLINATION = 0xc9,
77    JBT_REG_GAMMA1_BLUE_OFFSET = 0xca,
78
79    /* VGA */
80    JBT_REG_BLANK_CONTROL = 0xcf,
81    JBT_REG_BLANK_TH_TV = 0xd0,
82    JBT_REG_CKV_ON_OFF = 0xd1,
83    JBT_REG_CKV_1_2 = 0xd2,
84    JBT_REG_OEV_TIMING = 0xd3,
85    JBT_REG_ASW_TIMING_1 = 0xd4,
86    JBT_REG_ASW_TIMING_2 = 0xd5,
87
88    /* QVGA */
89    JBT_REG_BLANK_CONTROL_QVGA = 0xd6,
90    JBT_REG_BLANK_TH_TV_QVGA = 0xd7,
91    JBT_REG_CKV_ON_OFF_QVGA = 0xd8,
92    JBT_REG_CKV_1_2_QVGA = 0xd9,
93    JBT_REG_OEV_TIMING_QVGA = 0xde,
94    JBT_REG_ASW_TIMING_1_QVGA = 0xdf,
95    JBT_REG_ASW_TIMING_2_QVGA = 0xe0,
96
97
98    JBT_REG_HCLOCK_VGA = 0xec,
99    JBT_REG_HCLOCK_QVGA = 0xed,
100
101};
102
103enum jbt_resolution {
104    JBT_RESOLUTION_VGA,
105    JBT_RESOLUTION_QVGA,
106};
107
108enum jbt_power_mode {
109    JBT_POWER_MODE_DEEP_STANDBY,
110    JBT_POWER_MODE_SLEEP,
111    JBT_POWER_MODE_NORMAL,
112};
113
114static const char *jbt_power_mode_names[] = {
115    [JBT_POWER_MODE_DEEP_STANDBY] = "deep-standby",
116    [JBT_POWER_MODE_SLEEP] = "sleep",
117    [JBT_POWER_MODE_NORMAL] = "normal",
118};
119
120static const char *jbt_resolution_names[] = {
121    [JBT_RESOLUTION_VGA] = "vga",
122    [JBT_RESOLUTION_QVGA] = "qvga",
123};
124
125struct jbt_info {
126    struct mutex lock; /* protects this structure */
127    enum jbt_resolution resolution;
128    enum jbt_power_mode power_mode;
129    enum jbt_power_mode suspend_mode;
130    int suspended;
131    struct spi_device *spi_dev;
132    struct lcd_device *lcd_dev;
133    unsigned long last_sleep;
134    struct delayed_work blank_work;
135    int blank_mode;
136    u16 tx_buf[4];
137    u16 reg_cache[0xEE];
138};
139
140#define JBT_COMMAND 0x000
141#define JBT_DATA 0x100
142
143static int jbt_reg_write_nodata(struct jbt_info *jbt, u8 reg)
144{
145    int rc;
146
147    jbt->tx_buf[0] = JBT_COMMAND | reg;
148    rc = spi_write(jbt->spi_dev, (u8 *)jbt->tx_buf,
149                   1*sizeof(u16));
150    if (rc == 0)
151        jbt->reg_cache[reg] = 0;
152    else
153        dev_err(&jbt->spi_dev->dev, "jbt_reg_write_nodata spi_write ret %d\n",
154               rc);
155
156    return rc;
157}
158
159
160static int jbt_reg_write(struct jbt_info *jbt, u8 reg, u8 data)
161{
162    int rc;
163
164    jbt->tx_buf[0] = JBT_COMMAND | reg;
165    jbt->tx_buf[1] = JBT_DATA | data;
166    rc = spi_write(jbt->spi_dev, (u8 *)jbt->tx_buf,
167                   2*sizeof(u16));
168    if (rc == 0)
169        jbt->reg_cache[reg] = data;
170    else
171        dev_err(&jbt->spi_dev->dev, "jbt_reg_write spi_write ret %d\n", rc);
172
173    return rc;
174}
175
176static int jbt_reg_write16(struct jbt_info *jbt, u8 reg, u16 data)
177{
178    int rc;
179
180    jbt->tx_buf[0] = JBT_COMMAND | reg;
181    jbt->tx_buf[1] = JBT_DATA | (data >> 8);
182    jbt->tx_buf[2] = JBT_DATA | (data & 0xff);
183
184    rc = spi_write(jbt->spi_dev, (u8 *)jbt->tx_buf,
185                   3*sizeof(u16));
186    if (rc == 0)
187        jbt->reg_cache[reg] = data;
188    else
189        dev_err(&jbt->spi_dev->dev, "jbt_reg_write16 spi_write ret %d\n", rc);
190
191    return rc;
192}
193
194static int jbt_init_regs(struct jbt_info *jbt)
195{
196    int rc;
197
198    dev_dbg(&jbt->spi_dev->dev, "entering %cVGA mode\n",
199            jbt->resolution == JBT_RESOLUTION_QVGA ? 'Q' : ' ');
200
201    rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE1, 0x01);
202    rc |= jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE2, 0x00);
203    rc |= jbt_reg_write(jbt, JBT_REG_RGB_FORMAT, 0x60);
204    rc |= jbt_reg_write(jbt, JBT_REG_DRIVE_SYSTEM, 0x10);
205    rc |= jbt_reg_write(jbt, JBT_REG_BOOSTER_OP, 0x56);
206    rc |= jbt_reg_write(jbt, JBT_REG_BOOSTER_MODE, 0x33);
207    rc |= jbt_reg_write(jbt, JBT_REG_BOOSTER_FREQ, 0x11);
208    rc |= jbt_reg_write(jbt, JBT_REG_OPAMP_SYSCLK, 0x02);
209    rc |= jbt_reg_write(jbt, JBT_REG_VSC_VOLTAGE, 0x2b);
210    rc |= jbt_reg_write(jbt, JBT_REG_VCOM_VOLTAGE, 0x40);
211    rc |= jbt_reg_write(jbt, JBT_REG_EXT_DISPL, 0x03);
212    rc |= jbt_reg_write(jbt, JBT_REG_DCCLK_DCEV, 0x04);
213    /*
214     * default of 0x02 in JBT_REG_ASW_SLEW responsible for 72Hz requirement
215     * to avoid red / blue flicker
216     */
217    rc |= jbt_reg_write(jbt, JBT_REG_ASW_SLEW, 0x04 | (1 << 5));
218    rc |= jbt_reg_write(jbt, JBT_REG_DUMMY_DISPLAY, 0x00);
219
220    rc |= jbt_reg_write(jbt, JBT_REG_SLEEP_OUT_FR_A, 0x11);
221    rc |= jbt_reg_write(jbt, JBT_REG_SLEEP_OUT_FR_B, 0x11);
222    rc |= jbt_reg_write(jbt, JBT_REG_SLEEP_OUT_FR_C, 0x11);
223    rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_D, 0x2040);
224    rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_E, 0x60c0);
225    rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_F, 0x1020);
226    rc |= jbt_reg_write16(jbt, JBT_REG_SLEEP_IN_LCCNT_G, 0x60c0);
227
228    rc |= jbt_reg_write16(jbt, JBT_REG_GAMMA1_FINE_1, 0x5533);
229    rc |= jbt_reg_write(jbt, JBT_REG_GAMMA1_FINE_2, 0x00);
230    rc |= jbt_reg_write(jbt, JBT_REG_GAMMA1_INCLINATION, 0x00);
231    rc |= jbt_reg_write(jbt, JBT_REG_GAMMA1_BLUE_OFFSET, 0x00);
232
233    if (jbt->resolution != JBT_RESOLUTION_QVGA) {
234        rc |= jbt_reg_write16(jbt, JBT_REG_HCLOCK_VGA, 0x1f0);
235        rc |= jbt_reg_write(jbt, JBT_REG_BLANK_CONTROL, 0x02);
236        rc |= jbt_reg_write16(jbt, JBT_REG_BLANK_TH_TV, 0x0804);
237
238        rc |= jbt_reg_write(jbt, JBT_REG_CKV_ON_OFF, 0x01);
239        rc |= jbt_reg_write16(jbt, JBT_REG_CKV_1_2, 0x0000);
240
241        rc |= jbt_reg_write16(jbt, JBT_REG_OEV_TIMING, 0x0d0e);
242        rc |= jbt_reg_write16(jbt, JBT_REG_ASW_TIMING_1, 0x11a4);
243        rc |= jbt_reg_write(jbt, JBT_REG_ASW_TIMING_2, 0x0e);
244    } else {
245        rc |= jbt_reg_write16(jbt, JBT_REG_HCLOCK_QVGA, 0x00ff);
246        rc |= jbt_reg_write(jbt, JBT_REG_BLANK_CONTROL_QVGA, 0x02);
247        rc |= jbt_reg_write16(jbt, JBT_REG_BLANK_TH_TV_QVGA, 0x0804);
248
249        rc |= jbt_reg_write(jbt, JBT_REG_CKV_ON_OFF_QVGA, 0x01);
250        rc |= jbt_reg_write16(jbt, JBT_REG_CKV_1_2_QVGA, 0x0008);
251
252        rc |= jbt_reg_write16(jbt, JBT_REG_OEV_TIMING_QVGA, 0x050a);
253        rc |= jbt_reg_write16(jbt, JBT_REG_ASW_TIMING_1_QVGA, 0x0a19);
254        rc |= jbt_reg_write(jbt, JBT_REG_ASW_TIMING_2_QVGA, 0x0a);
255    }
256
257    return rc ? -EIO : 0;
258}
259
260static int standby_to_sleep(struct jbt_info *jbt)
261{
262    int rc;
263
264    /* three times command zero */
265    rc = jbt_reg_write_nodata(jbt, 0x00);
266    mdelay(1);
267    rc |= jbt_reg_write_nodata(jbt, 0x00);
268    mdelay(1);
269    rc |= jbt_reg_write_nodata(jbt, 0x00);
270    mdelay(1);
271
272    /* deep standby out */
273    rc |= jbt_reg_write(jbt, JBT_REG_POWER_ON_OFF, 0x11);
274    mdelay(1);
275    rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE, 0x28);
276
277    /* (re)initialize register set */
278    rc |= jbt_init_regs(jbt);
279
280    return rc ? -EIO : 0;
281}
282
283static int sleep_to_normal(struct jbt_info *jbt)
284{
285    int rc;
286
287    /* Make sure we are 120 ms after SLEEP_OUT */
288    if (time_before(jiffies, jbt->last_sleep))
289        mdelay(jiffies_to_msecs(jbt->last_sleep - jiffies));
290
291    if (jbt->resolution == JBT_RESOLUTION_VGA) {
292        /* RGB I/F on, RAM wirte off, QVGA through, SIGCON enable */
293        rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE, 0x80);
294
295        /* Quad mode off */
296        rc |= jbt_reg_write(jbt, JBT_REG_QUAD_RATE, 0x00);
297    } else {
298        /* RGB I/F on, RAM wirte off, QVGA through, SIGCON enable */
299        rc = jbt_reg_write(jbt, JBT_REG_DISPLAY_MODE, 0x81);
300
301        /* Quad mode on */
302        rc |= jbt_reg_write(jbt, JBT_REG_QUAD_RATE, 0x22);
303    }
304
305    /* AVDD on, XVDD on */
306    rc |= jbt_reg_write(jbt, JBT_REG_POWER_ON_OFF, 0x16);
307
308    /* Output control */
309    rc |= jbt_reg_write16(jbt, JBT_REG_OUTPUT_CONTROL, 0xfff9);
310
311    /* Turn on display */
312    rc |= jbt_reg_write_nodata(jbt, JBT_REG_DISPLAY_ON);
313
314    /* Sleep mode off */
315    rc |= jbt_reg_write_nodata(jbt, JBT_REG_SLEEP_OUT);
316    jbt->last_sleep = jiffies + msecs_to_jiffies(120);
317
318    /* Allow the booster and display controller to restart stably */
319    mdelay(5);
320
321    return rc ? -EIO : 0;
322}
323
324static int normal_to_sleep(struct jbt_info *jbt)
325{
326    int rc;
327
328    /* Make sure we are 120 ms after SLEEP_OUT */
329    while (time_before(jiffies, jbt->last_sleep))
330        cpu_relax();
331
332    rc = jbt_reg_write_nodata(jbt, JBT_REG_DISPLAY_OFF);
333    rc |= jbt_reg_write16(jbt, JBT_REG_OUTPUT_CONTROL, 0x8000 | 1 << 3);
334    rc |= jbt_reg_write_nodata(jbt, JBT_REG_SLEEP_IN);
335    jbt->last_sleep = jiffies + msecs_to_jiffies(120);
336
337    /* Allow the internal circuits to stop automatically */
338    mdelay(5);
339
340    return rc ? -EIO : 0;
341}
342
343static int sleep_to_standby(struct jbt_info *jbt)
344{
345    return jbt_reg_write(jbt, JBT_REG_POWER_ON_OFF, 0x00);
346}
347
348int jbt6k74_enter_power_mode(struct jbt_info *jbt, enum jbt_power_mode new_mode)
349{
350    struct jbt6k74_platform_data *pdata = jbt->spi_dev->dev.platform_data;
351    int rc = -EINVAL;
352
353    dev_dbg(&jbt->spi_dev->dev, "entering (old_state=%s, new_state=%s)\n",
354            jbt_power_mode_names[jbt->power_mode],
355            jbt_power_mode_names[new_mode]);
356
357    mutex_lock(&jbt->lock);
358
359    if (jbt->suspended) {
360        switch (new_mode) {
361        case JBT_POWER_MODE_DEEP_STANDBY:
362        case JBT_POWER_MODE_SLEEP:
363        case JBT_POWER_MODE_NORMAL:
364            rc = 0;
365            jbt->suspend_mode = new_mode;
366            break;
367        default:
368            break;
369        }
370    } else if (new_mode == JBT_POWER_MODE_NORMAL &&
371               pdata->enable_pixel_clock) {
372        pdata->enable_pixel_clock(&jbt->spi_dev->dev, 1);
373    }
374
375    switch (jbt->power_mode) {
376    case JBT_POWER_MODE_DEEP_STANDBY:
377        switch (new_mode) {
378        case JBT_POWER_MODE_DEEP_STANDBY:
379            rc = 0;
380            break;
381        case JBT_POWER_MODE_SLEEP:
382            rc = standby_to_sleep(jbt);
383            break;
384        case JBT_POWER_MODE_NORMAL:
385            /* first transition into sleep */
386            rc = standby_to_sleep(jbt);
387            /* then transition into normal */
388            rc |= sleep_to_normal(jbt);
389            break;
390        }
391        break;
392    case JBT_POWER_MODE_SLEEP:
393        switch (new_mode) {
394        case JBT_POWER_MODE_SLEEP:
395            rc = 0;
396            break;
397        case JBT_POWER_MODE_DEEP_STANDBY:
398            rc = sleep_to_standby(jbt);
399            break;
400        case JBT_POWER_MODE_NORMAL:
401            rc = sleep_to_normal(jbt);
402            break;
403        }
404        break;
405    case JBT_POWER_MODE_NORMAL:
406        switch (new_mode) {
407        case JBT_POWER_MODE_NORMAL:
408            rc = 0;
409            break;
410        case JBT_POWER_MODE_DEEP_STANDBY:
411            /* first transition into sleep */
412            rc = normal_to_sleep(jbt);
413            /* then transition into deep standby */
414            rc |= sleep_to_standby(jbt);
415            break;
416        case JBT_POWER_MODE_SLEEP:
417            rc = normal_to_sleep(jbt);
418            break;
419        }
420    }
421
422    if (rc == 0) {
423        jbt->power_mode = new_mode;
424        if (new_mode != JBT_POWER_MODE_NORMAL &&
425            pdata->enable_pixel_clock)
426            pdata->enable_pixel_clock(&jbt->spi_dev->dev, 0);
427    } else {
428        dev_err(&jbt->spi_dev->dev, "Failed enter state '%s')\n",
429                jbt_power_mode_names[new_mode]);
430    }
431
432    mutex_unlock(&jbt->lock);
433
434    return rc;
435}
436EXPORT_SYMBOL_GPL(jbt6k74_enter_power_mode);
437
438int jbt6k74_set_resolution(struct jbt_info *jbt, enum jbt_resolution new_resolution) {
439    int rc = 0;
440    enum jbt_resolution old_resolution;
441
442    if (new_resolution != JBT_RESOLUTION_VGA &&
443        new_resolution != JBT_RESOLUTION_QVGA)
444        return -EINVAL;
445
446    mutex_lock(&jbt->lock);
447
448    if (jbt->resolution == new_resolution)
449        goto out_unlock;
450
451    old_resolution = jbt->resolution;
452    jbt->resolution = new_resolution;
453
454    if (jbt->power_mode == JBT_POWER_MODE_NORMAL) {
455
456        /* first transition into sleep */
457        rc = normal_to_sleep(jbt);
458        /* second transition into deep standby */
459/* rc |= sleep_to_standby(jbt);*/
460        /* third transition into sleep */
461/* rc |= standby_to_sleep(jbt);*/
462        /* fourth transition into normal */
463        rc |= sleep_to_normal(jbt);
464
465        if (rc) {
466            jbt->resolution = old_resolution;
467            dev_err(&jbt->spi_dev->dev, "Failed to set resolution '%s')\n",
468                    jbt_resolution_names[new_resolution]);
469        }
470    }
471
472out_unlock:
473    mutex_unlock(&jbt->lock);
474
475    return rc;
476}
477EXPORT_SYMBOL_GPL(jbt6k74_set_resolution);
478
479static ssize_t resolution_read(struct device *dev, struct device_attribute *attr,
480              char *buf)
481{
482    struct jbt_info *jbt = dev_get_drvdata(dev);
483
484    if (jbt->resolution >= ARRAY_SIZE(jbt_resolution_names))
485        return -EIO;
486
487    return sprintf(buf, "%s\n", jbt_resolution_names[jbt->resolution]);
488}
489
490static ssize_t resolution_write(struct device *dev, struct device_attribute *attr,
491               const char *buf, size_t count)
492{
493    struct jbt_info *jbt = dev_get_drvdata(dev);
494    int i, rc;
495
496    for (i = 0; i < ARRAY_SIZE(jbt_resolution_names); i++) {
497        if (!strncmp(buf, jbt_resolution_names[i],
498               strlen(jbt_resolution_names[i]))) {
499            rc = jbt6k74_set_resolution(jbt, i);
500            if (rc)
501                return rc;
502            return count;
503        }
504    }
505
506    return -EINVAL;
507}
508
509static DEVICE_ATTR(resolution, 0644, resolution_read, resolution_write);
510
511static int reg_by_string(const char *name)
512{
513    if (!strcmp(name, "gamma_fine1"))
514        return JBT_REG_GAMMA1_FINE_1;
515    else if (!strcmp(name, "gamma_fine2"))
516        return JBT_REG_GAMMA1_FINE_2;
517    else if (!strcmp(name, "gamma_inclination"))
518        return JBT_REG_GAMMA1_INCLINATION;
519    else
520        return JBT_REG_GAMMA1_BLUE_OFFSET;
521}
522
523static ssize_t gamma_read(struct device *dev, struct device_attribute *attr,
524              char *buf)
525{
526    struct jbt_info *jbt = dev_get_drvdata(dev);
527    int reg = reg_by_string(attr->attr.name);
528    u16 val;
529
530    mutex_lock(&jbt->lock);
531    val = jbt->reg_cache[reg];
532        mutex_unlock(&jbt->lock);
533
534    return sprintf(buf, "0x%04x\n", val);
535}
536
537static ssize_t gamma_write(struct device *dev, struct device_attribute *attr,
538               const char *buf, size_t count)
539{
540    struct jbt_info *jbt = dev_get_drvdata(dev);
541    int reg = reg_by_string(attr->attr.name);
542    unsigned long val = simple_strtoul(buf, NULL, 10);
543
544    dev_info(dev, "writing gama %lu\n", val & 0xff);
545
546    mutex_lock(&jbt->lock);
547    jbt_reg_write(jbt, reg, val & 0xff);
548    mutex_unlock(&jbt->lock);
549
550    return count;
551}
552
553static ssize_t reset_write(struct device *dev, struct device_attribute *attr,
554               const char *buf, size_t count)
555{
556    int rc;
557    struct jbt_info *jbt = dev_get_drvdata(dev);
558    struct jbt6k74_platform_data *pdata = jbt->spi_dev->dev.platform_data;
559
560    dev_info(dev, "reset\n");
561
562    mutex_lock(&jbt->lock);
563
564    /* hard reset the jbt6k74 */
565    (pdata->reset)(0, 0);
566    mdelay(1);
567    (pdata->reset)(0, 1);
568    mdelay(120);
569
570    rc = jbt_reg_write_nodata(jbt, 0x01);
571    if (rc < 0)
572        dev_err(&jbt->spi_dev->dev, "cannot soft reset\n");
573    mdelay(120);
574
575    mutex_unlock(&jbt->lock);
576
577    jbt6k74_enter_power_mode(jbt, jbt->power_mode);
578
579    return count;
580}
581
582static DEVICE_ATTR(gamma_fine1, 0644, gamma_read, gamma_write);
583static DEVICE_ATTR(gamma_fine2, 0644, gamma_read, gamma_write);
584static DEVICE_ATTR(gamma_inclination, 0644, gamma_read, gamma_write);
585static DEVICE_ATTR(gamma_blue_offset, 0644, gamma_read, gamma_write);
586static DEVICE_ATTR(reset, 0600, NULL, reset_write);
587
588static struct attribute *jbt_sysfs_entries[] = {
589    &dev_attr_resolution.attr,
590    &dev_attr_gamma_fine1.attr,
591    &dev_attr_gamma_fine2.attr,
592    &dev_attr_gamma_inclination.attr,
593    &dev_attr_gamma_blue_offset.attr,
594    &dev_attr_reset.attr,
595    NULL,
596};
597
598static struct attribute_group jbt_attr_group = {
599    .name = NULL,
600    .attrs = jbt_sysfs_entries,
601};
602
603/* FIXME: This in an ugly hack to delay display blanking.
604  When the jbt is in sleep mode it displays an all white screen and thus one
605  will a see a short flash.
606  By delaying the blanking we will give the backlight a chance to turn off and
607  thus avoid getting the flash */
608static void jbt_blank_worker(struct work_struct *work) {
609    struct jbt_info *jbt = container_of(work, struct jbt_info,
610                        blank_work.work);
611
612    switch (jbt->blank_mode) {
613    case FB_BLANK_NORMAL:
614        jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_SLEEP);
615        break;
616    case FB_BLANK_POWERDOWN:
617        jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_DEEP_STANDBY);
618        break;
619    default:
620        break;
621    }
622}
623
624static int jbt6k74_set_mode(struct lcd_device *ld, struct fb_videomode *m) {
625    int rc = -EINVAL;
626    struct jbt_info *jbt = dev_get_drvdata(&ld->dev);
627
628    if (m->xres == 240 && m->yres == 320) {
629        rc = jbt6k74_set_resolution(jbt, JBT_RESOLUTION_QVGA);
630    } else if (m->xres == 480 && m->yres == 640) {
631        rc = jbt6k74_set_resolution(jbt, JBT_RESOLUTION_VGA);
632    } else {
633        dev_err(&jbt->spi_dev->dev, "Unknown resolution. Entering sleep mode.\n");
634        jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_SLEEP);
635    }
636
637    return rc;
638}
639
640static int jbt6k74_set_power(struct lcd_device *ld, int power) {
641    int rc = -EINVAL;
642    struct jbt_info *jbt = dev_get_drvdata(&ld->dev);
643
644    jbt->blank_mode = power;
645    cancel_rearming_delayed_work(&jbt->blank_work);
646
647    switch (power) {
648    case FB_BLANK_UNBLANK:
649        dev_dbg(&jbt->spi_dev->dev, "unblank\n");
650        rc = jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_NORMAL);
651        break;
652    case FB_BLANK_NORMAL:
653        dev_dbg(&jbt->spi_dev->dev, "blank\n");
654        rc = schedule_delayed_work(&jbt->blank_work, HZ);
655        break;
656    case FB_BLANK_POWERDOWN:
657        dev_dbg(&jbt->spi_dev->dev, "powerdown\n");
658        rc = schedule_delayed_work(&jbt->blank_work, HZ);
659        break;
660    default:
661        break;
662    }
663
664    return rc;
665}
666
667static int jbt6k74_get_power(struct lcd_device *ld) {
668    struct jbt_info *jbt = dev_get_drvdata(&ld->dev);
669
670    switch (jbt->power_mode) {
671        case JBT_POWER_MODE_NORMAL:
672        return FB_BLANK_UNBLANK;
673        case JBT_POWER_MODE_SLEEP:
674        return FB_BLANK_NORMAL;
675        default:
676        return JBT_POWER_MODE_DEEP_STANDBY;
677    }
678}
679
680struct lcd_ops jbt6k74_lcd_ops = {
681    .set_power = jbt6k74_set_power,
682    .get_power = jbt6k74_get_power,
683    .set_mode = jbt6k74_set_mode,
684};
685
686/* linux device model infrastructure */
687
688static int __devinit jbt_probe(struct spi_device *spi)
689{
690    int rc;
691    struct jbt_info *jbt;
692    struct jbt6k74_platform_data *pdata = spi->dev.platform_data;
693
694    /* the controller doesn't have a MISO pin; we can't do detection */
695
696    spi->mode = SPI_CPOL | SPI_CPHA;
697    spi->bits_per_word = 9;
698
699    rc = spi_setup(spi);
700    if (rc < 0) {
701        dev_err(&spi->dev,
702            "error during spi_setup of jbt6k74 driver\n");
703        return rc;
704    }
705
706    jbt = kzalloc(sizeof(*jbt), GFP_KERNEL);
707    if (!jbt)
708        return -ENOMEM;
709
710    jbt->spi_dev = spi;
711
712    jbt->lcd_dev = lcd_device_register("jbt6k74-lcd", &spi->dev,
713                       jbt, &jbt6k74_lcd_ops);
714
715    if (IS_ERR(jbt->lcd_dev)) {
716        rc = PTR_ERR(jbt->lcd_dev);
717        goto err_free_drvdata;
718    }
719
720    INIT_DELAYED_WORK(&jbt->blank_work, jbt_blank_worker);
721
722    jbt->resolution = JBT_RESOLUTION_VGA;
723    jbt->power_mode = JBT_POWER_MODE_DEEP_STANDBY;
724    jbt->last_sleep = jiffies + msecs_to_jiffies(120);
725    mutex_init(&jbt->lock);
726
727    dev_set_drvdata(&spi->dev, jbt);
728
729    rc = jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_NORMAL);
730    if (rc < 0) {
731        dev_err(&spi->dev, "cannot enter NORMAL state\n");
732        goto err_unregister_lcd;
733    }
734
735    rc = sysfs_create_group(&spi->dev.kobj, &jbt_attr_group);
736    if (rc < 0) {
737        dev_err(&spi->dev, "cannot create sysfs group\n");
738        goto err_standby;
739    }
740
741    if (pdata->probe_completed)
742        (pdata->probe_completed)(&spi->dev);
743
744    return 0;
745
746err_standby:
747    jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_DEEP_STANDBY);
748err_unregister_lcd:
749    lcd_device_unregister(jbt->lcd_dev);
750err_free_drvdata:
751    dev_set_drvdata(&spi->dev, NULL);
752    kfree(jbt);
753
754    return rc;
755}
756
757static int __devexit jbt_remove(struct spi_device *spi)
758{
759    struct jbt_info *jbt = dev_get_drvdata(&spi->dev);
760
761    /* We don't want to switch off the display in case the user
762     * accidentially unloads the module (whose use count normally is 0) */
763    jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_NORMAL);
764
765    sysfs_remove_group(&spi->dev.kobj, &jbt_attr_group);
766    dev_set_drvdata(&spi->dev, NULL);
767
768    lcd_device_unregister(jbt->lcd_dev);
769
770    kfree(jbt);
771
772    return 0;
773}
774
775#ifdef CONFIG_PM
776static int jbt_suspend(struct spi_device *spi, pm_message_t state)
777{
778    struct jbt_info *jbt = dev_get_drvdata(&spi->dev);
779
780    jbt->suspend_mode = jbt->power_mode;
781
782    jbt6k74_enter_power_mode(jbt, JBT_POWER_MODE_DEEP_STANDBY);
783    jbt->suspended = 1;
784
785    dev_info(&spi->dev, "suspended\n");
786
787    return 0;
788}
789
790int jbt6k74_resume(struct spi_device *spi)
791{
792    struct jbt_info *jbt = dev_get_drvdata(&spi->dev);
793
794    jbt->suspended = 0;
795    jbt6k74_enter_power_mode(jbt, jbt->suspend_mode);
796
797    dev_info(&spi->dev, "resumed\n");
798
799    return 0;
800}
801EXPORT_SYMBOL_GPL(jbt6k74_resume);
802
803#else
804#define jbt_suspend NULL
805#define jbt6k74_resume NULL
806#endif
807
808static struct spi_driver jbt6k74_driver = {
809    .driver = {
810        .name = "jbt6k74",
811        .owner = THIS_MODULE,
812    },
813
814    .probe = jbt_probe,
815    .remove = __devexit_p(jbt_remove),
816    .suspend = jbt_suspend,
817    .resume = jbt6k74_resume,
818};
819
820static int __init jbt_init(void)
821{
822    return spi_register_driver(&jbt6k74_driver);
823}
824
825static void __exit jbt_exit(void)
826{
827    spi_unregister_driver(&jbt6k74_driver);
828}
829
830MODULE_DESCRIPTION("SPI driver for tpo JBT6K74-AS LCM control interface");
831MODULE_AUTHOR("Harald Welte <laforge@openmoko.org>");
832MODULE_LICENSE("GPL");
833
834module_init(jbt_init);
835module_exit(jbt_exit);
836

Archive Download this file



interactive