diff options
| author | Doug Zobel <dzobel1@motorola.com> | 2013-11-15 14:29:07 -0600 |
|---|---|---|
| committer | James Wylder <jwylder@motorola.com> | 2014-03-05 17:46:52 -0600 |
| commit | d2a782003a6047da120a33e6f8ee6fd33bb825d6 (patch) | |
| tree | 8d20bd4ecda62a06e98993c4108456bc1acb0d0b /drivers/video/omap2/displays/panel-minnow.c | |
| parent | 32fd2d36d2464056d4522a9c02797b7c2b2e884f (diff) | |
| download | olio-linux-3.10-d2a782003a6047da120a33e6f8ee6fd33bb825d6.tar.xz olio-linux-3.10-d2a782003a6047da120a33e6f8ee6fd33bb825d6.zip | |
CW integration and minnow bringup
* create minnow machine type
* create Android makefile
* add pre-commit syntax check
* enable -Werror
* Add drivers: CPCAP, TPS65xxx, m4sensorhub, atmxt, lm3535,
usb gadget, minnow display, TI 12xx wireless
Change-Id: I7962f5e1256715f2452aed5a62a4f2f2383d5046
Diffstat (limited to 'drivers/video/omap2/displays/panel-minnow.c')
| -rw-r--r-- | drivers/video/omap2/displays/panel-minnow.c | 1761 |
1 files changed, 1761 insertions, 0 deletions
diff --git a/drivers/video/omap2/displays/panel-minnow.c b/drivers/video/omap2/displays/panel-minnow.c new file mode 100644 index 00000000000..aa0a380a5da --- /dev/null +++ b/drivers/video/omap2/displays/panel-minnow.c @@ -0,0 +1,1761 @@ +/* + * Minnow DSI command mode panel + * + * Copyright (C) 2013 Motorola Mobility LLC. + * + * 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. + * + * 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, see <http://www.gnu.org/licenses/>. + */ + +/*#define DEBUG*/ + +#include <linux/module.h> +#include <linux/delay.h> +#include <linux/err.h> +#include <linux/jiffies.h> +#include <linux/sched.h> +#include <linux/backlight.h> +#include <linux/fb.h> +#include <linux/interrupt.h> +#include <linux/gpio.h> +#include <linux/of_gpio.h> +#include <linux/workqueue.h> +#include <linux/slab.h> +#include <linux/mutex.h> + +#include <video/omapdss.h> +#include <video/omap-panel-data.h> +#include <video/mipi_display.h> + +/* DSI Virtual channel. Hardcoded for now. */ +#define TCH 0 + +#define DCS_READ_NUM_ERRORS 0x05 +#define DCS_BRIGHTNESS 0x51 +#define DCS_CTRL_DISPLAY 0x53 +#define DCS_WRITE_CABC 0x55 +#define DCS_READ_CABC 0x56 +#define DCS_GET_ID1 0xda +#define DCS_GET_ID2 0xdb +#define DCS_GET_ID3 0xdc + +enum minnow_panel_id { + MINNOW_PANEL_CM_220X176, + MINNOW_PANEL_CM_220X220, + MINNOW_PANEL_MAX +}; + +/* Panel Initialize DSI DCS command buffer description: + * it uses compact DCS command buffer to store all DCS commands, the first + * byte of each command is the command length in byte + */ +static u8 panel_init_220x176[] = { +/*n, data_0, data_1 ... data_n-1*/ + 3, 0xF0, 0x5A, 0x5A, + 3, 0xF1, 0x5A, 0x5A, +18, 0xF2, 0x16, 0xDC, 0x03, 0x28, 0x28, 0x10, 0x00, 0x60, 0xF8, 0x00, 0x07, 0x02, 0x00, 0x00, 0xDC, 0x28, 0x28, +15, 0xF4, 0x0A, 0x00, 0x00, 0x00, 0x77, 0x7F, 0x07, 0x22, 0x2A, 0x43, 0x07, 0x2A, 0x43, 0x07, +11, 0xF5, 0x00, 0x50, 0x28, 0x00, 0x00, 0x09, 0x00, 0x00, 0x00, 0x00, +10, 0xF6, 0x07, 0x00, 0x07, 0x00, 0x0B, 0x04, 0x04, 0x04, 0x07, + 5, 0xF7, 0x00, 0x00, 0x00, 0x00, + 3, 0xF8, 0x44, 0x08, + 2, 0xF9, 0x04, +17, 0xFA, 0x0F, 0x0F, 0x1E, 0x23, 0x26, 0x2D, 0x21, 0x2B, 0x33, 0x32, 0x2E, 0x00, 0x00, 0x00, 0x00, 0x00, +17, 0xFB, 0x0F, 0x0F, 0x1E, 0x23, 0x26, 0x2D, 0x21, 0x2B, 0x33, 0x32, 0x2E, 0x00, 0x00, 0x00, 0x00, 0x00, + 2, 0xF9, 0x02, +17, 0xFA, 0x00, 0x00, 0x0A, 0x16, 0x1D, 0x27, 0x1C, 0x30, 0x38, 0x37, 0x32, 0x00, 0x00, 0x00, 0x00, 0x00, +17, 0xFB, 0x00, 0x00, 0x0A, 0x16, 0x1D, 0x27, 0x1C, 0x30, 0x38, 0x37, 0x32, 0x00, 0x00, 0x00, 0x00, 0x00, + 2, 0xF9, 0x01, +17, 0xFA, 0x00, 0x00, 0x13, 0x14, 0x19, 0x24, 0x1A, 0x31, 0x39, 0x38, 0x33, 0x00, 0x00, 0x00, 0x00, 0x00, +17, 0xFB, 0x00, 0x00, 0x13, 0x14, 0x19, 0x24, 0x1A, 0x31, 0x39, 0x38, 0x33, 0x00, 0x00, 0x00, 0x00, 0x00, + 3, 0xF0, 0x00, 0x00, + 3, 0xF1, 0x00, 0x00, + 2, 0x36, 0xD0, + 2, 0x3A, 0x06 +}; + +static u8 panel_init_220x220[] = { +/*n, data_0, data_1 ... data_n-1*/ + 3, 0xF0, 0x5A, 0x5A, + 3, 0xF1, 0x5A, 0x5A, +18, 0xF2, 0x1C, 0xDC, 0x03, 0x28, 0x28, 0x10, 0x00, 0x60, 0xF8, 0x00, 0x07, 0x02, 0x00, 0x00, 0xDC, 0x28, 0x28, +15, 0xF4, 0x0A, 0x00, 0x00, 0x00, 0x77, 0x7F, 0x07, 0x22, 0x2A, 0x43, 0x07, 0x2A, 0x43, 0x07, +11, 0xF5, 0x00, 0x50, 0x28, 0x00, 0x00, 0x09, 0x00, 0x00, 0x01, 0x01, +10, 0xF6, 0x07, 0x00, 0x07, 0x00, 0x0B, 0x04, 0x04, 0x04, 0x07, + 5, 0xF7, 0x00, 0x00, 0x00, 0x00, + 3, 0xF8, 0x44, 0x02, + 2, 0xF9, 0x04, +17, 0xFA, 0x1E, 0x1E, 0x0D, 0x1D, 0x21, 0x2C, 0x23, 0x28, 0x2C, 0x28, 0x28, 0x00, 0x00, 0x00, 0x00, 0x00, +17, 0xFB, 0x1E, 0x1E, 0x0D, 0x1D, 0x21, 0x2C, 0x23, 0x28, 0x2C, 0x28, 0x28, 0x00, 0x00, 0x00, 0x00, 0x00, + 2, 0xF9, 0x02, +17, 0xFA, 0x19, 0x18, 0x08, 0x0F, 0x18, 0x26, 0x1E, 0x2C, 0x30, 0x2C, 0x2C, 0x00, 0x00, 0x00, 0x00, 0x00, +17, 0xFB, 0x19, 0x18, 0x08, 0x0F, 0x18, 0x26, 0x1E, 0x2C, 0x30, 0x2C, 0x2C, 0x00, 0x00, 0x00, 0x00, 0x00, + 2, 0xF9, 0x01, +17, 0xFA, 0x19, 0x19, 0x09, 0x0D, 0x12, 0x21, 0x1B, 0x2E, 0x31, 0x2E, 0x2E, 0x00, 0x00, 0x00, 0x00, 0x00, +17, 0xFB, 0x19, 0x19, 0x09, 0x0D, 0x12, 0x21, 0x1B, 0x2E, 0x31, 0x2E, 0x2E, 0x00, 0x00, 0x00, 0x00, 0x00, + 3, 0xF0, 0x00, 0x00, + 3, 0xF1, 0x00, 0x00, + 2, 0x36, 0xD0, + 2, 0x3A, 0x06 +}; + +struct minnow_panel_attr { + int xres; + int yres; + int rate; + int xoffset; + int yoffset; + int init_cmd_count; + u8 *init_cmd; +}; + +#define INIT_CMD(buf) .init_cmd_count = sizeof(buf), .init_cmd = (buf) +static struct minnow_panel_attr panel_attr_table[MINNOW_PANEL_MAX] = { + [MINNOW_PANEL_CM_220X176] = { + .xres = 220, + .yres = 176, + .rate = 60, + .xoffset = 0x32, + .yoffset = 0, + INIT_CMD(panel_init_220x176) + }, + [MINNOW_PANEL_CM_220X220] = { + .xres = 220, + .yres = 220, + .rate = 60, + .xoffset = 0x32, + .yoffset = 0x4, + INIT_CMD(panel_init_220x220) + }, +}; + +static irqreturn_t minnow_panel_te_isr(int irq, void *data); +static void minnow_panel_te_timeout_work_callback(struct work_struct *work); +static int _minnow_panel_enable_te(struct omap_dss_device *dssdev, bool enable); + +static int minnow_panel_reset(struct omap_dss_device *dssdev); + +struct minnow_panel_data { + struct mutex lock; /* mutex */ + + struct backlight_device *bldev; + + unsigned long hw_guard_end; /* next value of jiffies when we can + * issue the next sleep in/out command + */ + unsigned long hw_guard_wait; /* max guard time in jiffies */ + + struct omap_dss_device *dssdev; + + /* panel HW configuration from DT or platform data */ + int reset_gpio; + int ext_te_gpio; + + bool use_dsi_backlight; + + struct omap_dsi_pin_config pin_config; + struct omap_dss_dsi_config dsi_config; + + u8 *init_cmd_data; + int init_cmd_count; + int x_offset; + int y_offset; + + /* runtime variables */ + bool enabled; + + bool te_enabled; + + atomic_t do_update; + int channel; + + struct delayed_work te_timeout_work; + + unsigned cabc_mode; + + bool intro_printed; + + struct workqueue_struct *workqueue; + + struct delayed_work esd_work; + unsigned esd_interval; + + bool ulps_enabled; + unsigned ulps_timeout; + struct delayed_work ulps_work; +}; + +static void minnow_panel_esd_work(struct work_struct *work); +static void minnow_panel_ulps_work(struct work_struct *work); + +static void hw_guard_start(struct minnow_panel_data *mpd, int guard_msec) +{ + mpd->hw_guard_wait = msecs_to_jiffies(guard_msec); + mpd->hw_guard_end = jiffies + mpd->hw_guard_wait; +} + +static void hw_guard_wait(struct minnow_panel_data *mpd) +{ + unsigned long wait = mpd->hw_guard_end - jiffies; + + if ((long)wait > 0 && wait <= mpd->hw_guard_wait) { + set_current_state(TASK_UNINTERRUPTIBLE); + schedule_timeout(wait); + } +} + +static int minnow_panel_dcs_read_1(struct minnow_panel_data *mpd, u8 dcs_cmd, u8 *data) +{ + int r; + u8 buf[1]; + + r = dsi_vc_dcs_read(mpd->dssdev, mpd->channel, dcs_cmd, buf, 1); + + if (r < 0) + return r; + + *data = buf[0]; + + return 0; +} + +static int minnow_panel_dcs_write_0(struct minnow_panel_data *mpd, u8 dcs_cmd) +{ + return dsi_vc_dcs_write(mpd->dssdev, mpd->channel, &dcs_cmd, 1); +} + +static int minnow_panel_dcs_write_1(struct minnow_panel_data *mpd, u8 dcs_cmd, u8 param) +{ + u8 buf[2]; + buf[0] = dcs_cmd; + buf[1] = param; + return dsi_vc_dcs_write(mpd->dssdev, mpd->channel, buf, 2); +} + +static int minnow_panel_sleep_in(struct minnow_panel_data *mpd) + +{ + u8 cmd; + int r; + + hw_guard_wait(mpd); + + cmd = MIPI_DCS_ENTER_SLEEP_MODE; + r = dsi_vc_dcs_write_nosync(mpd->dssdev, mpd->channel, &cmd, 1); + if (r) + return r; + + hw_guard_start(mpd, 120); + + msleep(10); + + return 0; +} + +static int minnow_panel_sleep_out(struct minnow_panel_data *mpd) +{ + int r; + + hw_guard_wait(mpd); + + r = minnow_panel_dcs_write_0(mpd, MIPI_DCS_EXIT_SLEEP_MODE); + if (r) + return r; + + hw_guard_start(mpd, 120); + + msleep(10); + + return 0; +} + +static int minnow_panel_get_id(struct minnow_panel_data *mpd, u8 *id1, u8 *id2, u8 *id3) +{ + int r; + + r = minnow_panel_dcs_read_1(mpd, DCS_GET_ID1, id1); + if (r) + return r; + r = minnow_panel_dcs_read_1(mpd, DCS_GET_ID2, id2); + if (r) + return r; + r = minnow_panel_dcs_read_1(mpd, DCS_GET_ID3, id3); + if (r) + return r; + + return 0; +} + +static int _minnow_panel_init(struct minnow_panel_data *mpd) +{ + int i, r; + for (i = 0; i < mpd->init_cmd_count; ) { + u8 *data = mpd->init_cmd_data + i; + r = *data++; + i += r + 1; + if (i > mpd->init_cmd_count) { + dev_err(&mpd->dssdev->dev, "Invalid init command data selected!\n"); + return -EINVAL; + } + r = dsi_vc_dcs_write(mpd->dssdev, mpd->channel, data, r); + if (r) + break; + } + return r; +} + +static int minnow_panel_set_update_window(struct minnow_panel_data *mpd, + u16 x, u16 y, u16 w, u16 h) +{ + int r; + u16 x1 = x + mpd->x_offset; + u16 x2 = x + mpd->x_offset + w - 1; + u16 y1 = y + mpd->y_offset; + u16 y2 = y + mpd->y_offset + h - 1; + + u8 buf[5]; + buf[0] = MIPI_DCS_SET_COLUMN_ADDRESS; + buf[1] = (x1 >> 8) & 0xff; + buf[2] = (x1 >> 0) & 0xff; + buf[3] = (x2 >> 8) & 0xff; + buf[4] = (x2 >> 0) & 0xff; + + r = dsi_vc_dcs_write_nosync(mpd->dssdev, mpd->channel, buf, sizeof(buf)); + if (r) + return r; + + buf[0] = MIPI_DCS_SET_PAGE_ADDRESS; + buf[1] = (y1 >> 8) & 0xff; + buf[2] = (y1 >> 0) & 0xff; + buf[3] = (y2 >> 8) & 0xff; + buf[4] = (y2 >> 0) & 0xff; + + r = dsi_vc_dcs_write_nosync(mpd->dssdev, mpd->channel, buf, sizeof(buf)); + if (r) + return r; + + dsi_vc_send_bta_sync(mpd->dssdev, mpd->channel); + + return r; +} + +static void minnow_panel_queue_esd_work(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + if (mpd->esd_interval > 0) + queue_delayed_work(mpd->workqueue, &mpd->esd_work, + msecs_to_jiffies(mpd->esd_interval)); +} + +static void minnow_panel_cancel_esd_work(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + cancel_delayed_work(&mpd->esd_work); +} + +static void minnow_panel_queue_ulps_work(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + if (mpd->ulps_timeout > 0) + queue_delayed_work(mpd->workqueue, &mpd->ulps_work, + msecs_to_jiffies(mpd->ulps_timeout)); +} + +static void minnow_panel_cancel_ulps_work(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + cancel_delayed_work(&mpd->ulps_work); +} + +static int minnow_panel_enter_ulps(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + if (mpd->ulps_enabled) + return 0; + + minnow_panel_cancel_ulps_work(dssdev); + + r = _minnow_panel_enable_te(dssdev, false); + if (r) + goto err; + + if (gpio_is_valid(mpd->ext_te_gpio)) + disable_irq(gpio_to_irq(mpd->ext_te_gpio)); + + omapdss_dsi_display_disable(dssdev, false, true); + + mpd->ulps_enabled = true; + + return 0; + +err: + dev_err(&dssdev->dev, "enter ULPS failed"); + minnow_panel_reset(dssdev); + + mpd->ulps_enabled = false; + + minnow_panel_queue_ulps_work(dssdev); + + return r; +} + +static int minnow_panel_exit_ulps(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + if (!mpd->ulps_enabled) + return 0; + + r = omapdss_dsi_display_enable(dssdev); + if (r) { + dev_err(&dssdev->dev, "failed to enable DSI\n"); + goto err1; + } + + omapdss_dsi_vc_enable_hs(dssdev, mpd->channel, true); + + r = _minnow_panel_enable_te(dssdev, true); + if (r) { + dev_err(&dssdev->dev, "failed to re-enable TE"); + goto err2; + } + + if (gpio_is_valid(mpd->ext_te_gpio)) + enable_irq(gpio_to_irq(mpd->ext_te_gpio)); + + minnow_panel_queue_ulps_work(dssdev); + + mpd->ulps_enabled = false; + + return 0; + +err2: + dev_err(&dssdev->dev, "failed to exit ULPS"); + + r = minnow_panel_reset(dssdev); + if (!r) { + if (gpio_is_valid(mpd->ext_te_gpio)) + enable_irq(gpio_to_irq(mpd->ext_te_gpio)); + mpd->ulps_enabled = false; + } +err1: + minnow_panel_queue_ulps_work(dssdev); + + return r; +} + +static int minnow_panel_wake_up(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + if (mpd->ulps_enabled) + return minnow_panel_exit_ulps(dssdev); + + minnow_panel_cancel_ulps_work(dssdev); + minnow_panel_queue_ulps_work(dssdev); + return 0; +} + +static int minnow_panel_bl_update_status(struct backlight_device *dev) +{ + struct omap_dss_device *dssdev = dev_get_drvdata(&dev->dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + int level; + + if (dev->props.fb_blank == FB_BLANK_UNBLANK && + dev->props.power == FB_BLANK_UNBLANK) + level = dev->props.brightness; + else + level = 0; + + dev_dbg(&dssdev->dev, "update brightness to %d\n", level); + + mutex_lock(&mpd->lock); + + if (mpd->enabled) { + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (!r) + r = minnow_panel_dcs_write_1(mpd, DCS_BRIGHTNESS, level); + + dsi_bus_unlock(dssdev); + } else { + r = 0; + } + + mutex_unlock(&mpd->lock); + + return r; +} + +static int minnow_panel_bl_get_intensity(struct backlight_device *dev) +{ + if (dev->props.fb_blank == FB_BLANK_UNBLANK && + dev->props.power == FB_BLANK_UNBLANK) + return dev->props.brightness; + + return 0; +} + +static const struct backlight_ops minnow_panel_bl_ops = { + .get_brightness = minnow_panel_bl_get_intensity, + .update_status = minnow_panel_bl_update_status, +}; + +static void minnow_panel_get_resolution(struct omap_dss_device *dssdev, + u16 *xres, u16 *yres) +{ + *xres = dssdev->panel.timings.x_res; + *yres = dssdev->panel.timings.y_res; +} + +static ssize_t minnow_panel_num_errors_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + u8 errors = 0; + int r; + + mutex_lock(&mpd->lock); + + if (mpd->enabled) { + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (!r) + r = minnow_panel_dcs_read_1(mpd, DCS_READ_NUM_ERRORS, &errors); + + dsi_bus_unlock(dssdev); + } else { + r = -ENODEV; + } + + mutex_unlock(&mpd->lock); + + if (r) + return r; + + return snprintf(buf, PAGE_SIZE, "%d\n", errors); +} + +static ssize_t minnow_panel_hw_revision_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + u8 id1, id2, id3; + int r; + + mutex_lock(&mpd->lock); + + if (mpd->enabled) { + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (!r) + r = minnow_panel_get_id(mpd, &id1, &id2, &id3); + + dsi_bus_unlock(dssdev); + } else { + r = -ENODEV; + } + + mutex_unlock(&mpd->lock); + + if (r) + return r; + + return snprintf(buf, PAGE_SIZE, "%02x.%02x.%02x\n", id1, id2, id3); +} + +static const char *cabc_modes[] = { + "off", /* used also always when CABC is not supported */ + "ui", + "still-image", + "moving-image", +}; + +static ssize_t show_cabc_mode(struct device *dev, + struct device_attribute *attr, + char *buf) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + const char *mode_str; + int mode; + int len; + + mode = mpd->cabc_mode; + + mode_str = "unknown"; + if (mode >= 0 && mode < ARRAY_SIZE(cabc_modes)) + mode_str = cabc_modes[mode]; + len = snprintf(buf, PAGE_SIZE, "%s\n", mode_str); + + return len < PAGE_SIZE - 1 ? len : PAGE_SIZE - 1; +} + +static ssize_t store_cabc_mode(struct device *dev, + struct device_attribute *attr, + const char *buf, size_t count) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int i; + int r; + + for (i = 0; i < ARRAY_SIZE(cabc_modes); i++) { + if (sysfs_streq(cabc_modes[i], buf)) + break; + } + + if (i == ARRAY_SIZE(cabc_modes)) + return -EINVAL; + + mutex_lock(&mpd->lock); + + if (mpd->enabled) { + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (r) + goto err; + + r = minnow_panel_dcs_write_1(mpd, DCS_WRITE_CABC, i); + if (r) + goto err; + + dsi_bus_unlock(dssdev); + } + + mpd->cabc_mode = i; + + mutex_unlock(&mpd->lock); + + return count; +err: + dsi_bus_unlock(dssdev); + mutex_unlock(&mpd->lock); + return r; +} + +static ssize_t show_cabc_available_modes(struct device *dev, + struct device_attribute *attr, + char *buf) +{ + int len; + int i; + + for (i = 0, len = 0; + len < PAGE_SIZE && i < ARRAY_SIZE(cabc_modes); i++) + len += snprintf(&buf[len], PAGE_SIZE - len, "%s%s%s", + i ? " " : "", cabc_modes[i], + i == ARRAY_SIZE(cabc_modes) - 1 ? "\n" : ""); + + return len < PAGE_SIZE ? len : PAGE_SIZE - 1; +} + +static ssize_t minnow_panel_store_esd_interval(struct device *dev, + struct device_attribute *attr, + const char *buf, size_t count) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + unsigned long t; + int r; + + r = strict_strtoul(buf, 10, &t); + if (r) + return r; + + mutex_lock(&mpd->lock); + minnow_panel_cancel_esd_work(dssdev); + mpd->esd_interval = t; + if (mpd->enabled) + minnow_panel_queue_esd_work(dssdev); + mutex_unlock(&mpd->lock); + + return count; +} + +static ssize_t minnow_panel_show_esd_interval(struct device *dev, + struct device_attribute *attr, + char *buf) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + unsigned t; + + mutex_lock(&mpd->lock); + t = mpd->esd_interval; + mutex_unlock(&mpd->lock); + + return snprintf(buf, PAGE_SIZE, "%u\n", t); +} + +static ssize_t minnow_panel_store_ulps(struct device *dev, + struct device_attribute *attr, + const char *buf, size_t count) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + unsigned long t; + int r; + + r = strict_strtoul(buf, 10, &t); + if (r) + return r; + + mutex_lock(&mpd->lock); + + if (mpd->enabled) { + dsi_bus_lock(dssdev); + + if (t) + r = minnow_panel_enter_ulps(dssdev); + else + r = minnow_panel_wake_up(dssdev); + + dsi_bus_unlock(dssdev); + } + + mutex_unlock(&mpd->lock); + + if (r) + return r; + + return count; +} + +static ssize_t minnow_panel_show_ulps(struct device *dev, + struct device_attribute *attr, + char *buf) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + unsigned t; + + mutex_lock(&mpd->lock); + t = mpd->ulps_enabled; + mutex_unlock(&mpd->lock); + + return snprintf(buf, PAGE_SIZE, "%u\n", t); +} + +static ssize_t minnow_panel_store_ulps_timeout(struct device *dev, + struct device_attribute *attr, + const char *buf, size_t count) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + unsigned long t; + int r; + + r = strict_strtoul(buf, 10, &t); + if (r) + return r; + + mutex_lock(&mpd->lock); + mpd->ulps_timeout = t; + + if (mpd->enabled) { + /* minnow_panel_wake_up will restart the timer */ + dsi_bus_lock(dssdev); + r = minnow_panel_wake_up(dssdev); + dsi_bus_unlock(dssdev); + } + + mutex_unlock(&mpd->lock); + + if (r) + return r; + + return count; +} + +static ssize_t minnow_panel_show_ulps_timeout(struct device *dev, + struct device_attribute *attr, + char *buf) +{ + struct omap_dss_device *dssdev = to_dss_device(dev); + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + unsigned t; + + mutex_lock(&mpd->lock); + t = mpd->ulps_timeout; + mutex_unlock(&mpd->lock); + + return snprintf(buf, PAGE_SIZE, "%u\n", t); +} + +static DEVICE_ATTR(num_dsi_errors, S_IRUGO, minnow_panel_num_errors_show, NULL); +static DEVICE_ATTR(hw_revision, S_IRUGO, minnow_panel_hw_revision_show, NULL); +static DEVICE_ATTR(cabc_mode, S_IRUGO | S_IWUSR, + show_cabc_mode, store_cabc_mode); +static DEVICE_ATTR(cabc_available_modes, S_IRUGO, + show_cabc_available_modes, NULL); +static DEVICE_ATTR(esd_interval, S_IRUGO | S_IWUSR, + minnow_panel_show_esd_interval, minnow_panel_store_esd_interval); +static DEVICE_ATTR(ulps, S_IRUGO | S_IWUSR, + minnow_panel_show_ulps, minnow_panel_store_ulps); +static DEVICE_ATTR(ulps_timeout, S_IRUGO | S_IWUSR, + minnow_panel_show_ulps_timeout, minnow_panel_store_ulps_timeout); + +static struct attribute *minnow_panel_attrs[] = { + &dev_attr_num_dsi_errors.attr, + &dev_attr_hw_revision.attr, + &dev_attr_cabc_mode.attr, + &dev_attr_cabc_available_modes.attr, + &dev_attr_esd_interval.attr, + &dev_attr_ulps.attr, + &dev_attr_ulps_timeout.attr, + NULL, +}; + +static struct attribute_group minnow_panel_attr_group = { + .attrs = minnow_panel_attrs, +}; + +static void minnow_panel_hw_reset(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + if (!gpio_is_valid(mpd->reset_gpio)) + return; + + gpio_set_value(mpd->reset_gpio, 1); + udelay(10); + /* reset the panel */ + gpio_set_value(mpd->reset_gpio, 0); + /* assert reset */ + udelay(10); + gpio_set_value(mpd->reset_gpio, 1); + + /* wait after releasing reset */ + msleep(5); +} + +#define DEBUG_DT +#ifdef DEBUG_DT +#define DTINFO(fmt, ...) \ + printk(KERN_INFO "minnow-panel DT: " fmt, ## __VA_ARGS__) +#define DTINFO_PIXFMT(msg, pix) \ +{ char *fmt[4] = {"RGB888", "RGB666", "RGB666_PACKED", "RGB565"};\ + DTINFO(msg"%s\n", fmt[pix]);\ +} +#define DTINFO_ARRAY(msg, a, n, fmt, blen) \ +{ int i; char str[blen], *p = str;\ + for (i = 0; i < n; i++) {\ + sprintf(p, fmt, a[i]);\ + p += strlen(p); \ + } \ + DTINFO(msg"%s\n", str);\ +} +#else /* DEBUG_DT */ +#define DTINFO(fmt, ...) +#define DTINFO_PIXFMT(msg, pix) +#define DTINFO_ARRAY(msg, a, n, fmt, blen) +#endif + +static struct of_device_id minnow_panel_ids[] = { + { .compatible = "mot,minnow-panel-dsi-cm" }, + { /*sentinel*/ } +}; + +static int minnow_panel_dt_init(struct minnow_panel_data *mpd) +{ + u32 range[2], value = 0; + struct minnow_panel_attr *panel_attr; + struct device_node *dt_node; + + dt_node = of_find_matching_node(NULL, minnow_panel_ids); + if (dt_node == NULL) { + dev_err(&mpd->dssdev->dev, "No dt_node found!\n"); + return -ENODEV; + } + + if (of_property_read_u32(dt_node, "id_panel", &value) \ + || (value >= MINNOW_PANEL_MAX)) { + dev_err(&mpd->dssdev->dev, \ + "Invalid id_panel = %u!\n", value); + return -EINVAL; + } + DTINFO("id_panel = %d\n", value); + panel_attr = &panel_attr_table[value]; + mpd->init_cmd_data = panel_attr->init_cmd; + mpd->init_cmd_count = panel_attr->init_cmd_count; + mpd->x_offset = panel_attr->xoffset; + mpd->y_offset = panel_attr->yoffset; + mpd->dssdev->panel.timings.x_res = panel_attr->xres; + mpd->dssdev->panel.timings.y_res = panel_attr->yres; + mpd->dssdev->panel.timings.pixel_clock = DIV_ROUND_UP(panel_attr->xres\ + * panel_attr->yres * panel_attr->rate, 1000); + mpd->dssdev->panel.dsi_pix_fmt = OMAP_DSS_DSI_FMT_RGB888; + mpd->dssdev->caps = OMAP_DSS_DISPLAY_CAP_MANUAL_UPDATE | + OMAP_DSS_DISPLAY_CAP_TEAR_ELIM; + mpd->dsi_config.mode = OMAP_DSS_DSI_CMD_MODE; + mpd->dsi_config.pixel_format = OMAP_DSS_DSI_FMT_RGB888; + mpd->dsi_config.hs_clk_min = 90000000; + mpd->dsi_config.hs_clk_max = 150000000; + mpd->dsi_config.lp_clk_min = 7000000; + mpd->dsi_config.lp_clk_max = 9000000; + + mpd->reset_gpio = of_get_named_gpio(dt_node, "gpio_reset", 0); + DTINFO("gpio_reset = %d\n", mpd->reset_gpio); + mpd->ext_te_gpio = of_get_named_gpio(dt_node, "gpio_te", 0); + DTINFO("ext_te: gpio_te = %d\n", mpd->ext_te_gpio); + + mpd->esd_interval = 0; + if (!of_property_read_u32(dt_node, "esd_interval", &value)) { + mpd->esd_interval = value; + DTINFO("esd_interval = %d\n", mpd->esd_interval); + } + mpd->ulps_timeout = 0; + mpd->use_dsi_backlight = false; + + mpd->pin_config.num_pins = 4; + mpd->pin_config.pins[0] = 0; + mpd->pin_config.pins[1] = 1; + mpd->pin_config.pins[2] = 2; + mpd->pin_config.pins[3] = 3; + if (of_get_property(dt_node, "pins", &value)) { + u32 pins[OMAP_DSS_MAX_DSI_PINS]; + u32 num_pins = value / sizeof(u32); + if (!num_pins || (num_pins > OMAP_DSS_MAX_DSI_PINS)) { + dev_err(&mpd->dssdev->dev, \ + "Invalid DSI pins count = %u!\n", num_pins); + return -EINVAL; + } + value = 0; + if (!of_property_read_u32_array(dt_node, \ + "pins", pins, num_pins)) { + for (; value < num_pins; value++) { + if (pins[value] >= OMAP_DSS_MAX_DSI_PINS) + break; + mpd->pin_config.pins[value]\ + = pins[value]; + } + } + if (value < num_pins) { + dev_err(&mpd->dssdev->dev, \ + "Invalid DSI pins config!\n"); + return -EINVAL; + } + mpd->pin_config.num_pins = num_pins; + DTINFO("num_pins = %d\n", \ + mpd->pin_config.num_pins); + DTINFO_ARRAY("pins =", mpd->pin_config.pins,\ + mpd->pin_config.num_pins, " %u", 64); + } + + if (!of_property_read_u32(dt_node, "pixel_clock", &value)) { + if (value < mpd->dssdev->panel.timings.pixel_clock) { + dev_err(&mpd->dssdev->dev, \ + "Invalid pixel_clock = %u!\n", value); + return -EINVAL; + } + mpd->dssdev->panel.timings.pixel_clock = value; + DTINFO("pixel_clock = %u\n", \ + mpd->dssdev->panel.timings.pixel_clock); + } + + if (!of_property_read_u32(dt_node, "pixel_format", &value)) { + switch (value) { + case OMAP_DSS_DSI_FMT_RGB888: + case OMAP_DSS_DSI_FMT_RGB666: + case OMAP_DSS_DSI_FMT_RGB666_PACKED: + case OMAP_DSS_DSI_FMT_RGB565: + break; + default: + dev_err(&mpd->dssdev->dev, \ + "Invalid pixel_format = %u!\n", value); + return -EINVAL; + } + mpd->dssdev->panel.dsi_pix_fmt = \ + mpd->dsi_config.pixel_format = value; + DTINFO_PIXFMT("pixel_format = ", \ + mpd->dssdev->panel.dsi_pix_fmt); + } + + if (!of_property_read_u32_array(dt_node, "hs_clk", range, 2)) { + mpd->dsi_config.hs_clk_min = range[0]; + mpd->dsi_config.hs_clk_max = range[1]; + DTINFO("hs_clk_min = %lu, hs_clk_max = %lu\n", \ + mpd->dsi_config.hs_clk_min, \ + mpd->dsi_config.hs_clk_max); + } + + if (!of_property_read_u32_array(dt_node, "lp_clk", range, 2)) { + mpd->dsi_config.lp_clk_min = range[0]; + mpd->dsi_config.lp_clk_max = range[1]; + DTINFO("lp_clk_min = %lu, lp_clk_max = %lu\n", \ + mpd->dsi_config.lp_clk_min, \ + mpd->dsi_config.lp_clk_max); + } + + return 0; +} + +static int minnow_panel_probe(struct omap_dss_device *dssdev) +{ + struct backlight_properties props; + struct minnow_panel_data *mpd; + struct backlight_device *bldev = NULL; + int r; + + dev_dbg(&dssdev->dev, "probe\n"); + + mpd = devm_kzalloc(&dssdev->dev, sizeof(*mpd), GFP_KERNEL); + if (!mpd) + return -ENOMEM; + + dev_set_drvdata(&dssdev->dev, mpd); + mpd->dssdev = dssdev; + + r = minnow_panel_dt_init(mpd); + if (r) + return r; + + mutex_init(&mpd->lock); + + atomic_set(&mpd->do_update, 0); + + if (gpio_is_valid(mpd->reset_gpio)) { + r = devm_gpio_request_one(&dssdev->dev, mpd->reset_gpio, + GPIOF_OUT_INIT_HIGH, "minnow-panel reset"); + if (r) { + dev_err(&dssdev->dev, "failed to request reset gpio\n"); + return r; + } + } + + if (gpio_is_valid(mpd->ext_te_gpio)) { + r = devm_gpio_request_one(&dssdev->dev, mpd->ext_te_gpio, + GPIOF_IN, "minnow-panel irq"); + if (r) { + dev_err(&dssdev->dev, "failed to request ext_te gpio\n"); + return r; + } + + r = devm_request_irq(&dssdev->dev, gpio_to_irq(mpd->ext_te_gpio), + minnow_panel_te_isr, + IRQF_TRIGGER_RISING, + "minnow-panel vsync", dssdev); + + if (r) { + dev_err(&dssdev->dev, "IRQ request failed\n"); + return r; + } + + INIT_DEFERRABLE_WORK(&mpd->te_timeout_work, + minnow_panel_te_timeout_work_callback); + + dev_dbg(&dssdev->dev, "Using GPIO TE\n"); + } + + mpd->workqueue = create_singlethread_workqueue("minnow_panel_esd"); + if (mpd->workqueue == NULL) { + dev_err(&dssdev->dev, "can't create ESD workqueue\n"); + return -ENOMEM; + } + INIT_DEFERRABLE_WORK(&mpd->esd_work, minnow_panel_esd_work); + INIT_DELAYED_WORK(&mpd->ulps_work, minnow_panel_ulps_work); + + if (mpd->use_dsi_backlight) { + memset(&props, 0, sizeof(struct backlight_properties)); + props.max_brightness = 255; + + props.type = BACKLIGHT_RAW; + bldev = backlight_device_register(dev_name(&dssdev->dev), + &dssdev->dev, dssdev, &minnow_panel_bl_ops, &props); + if (IS_ERR(bldev)) { + r = PTR_ERR(bldev); + goto err_bl; + } + + mpd->bldev = bldev; + + bldev->props.fb_blank = FB_BLANK_UNBLANK; + bldev->props.power = FB_BLANK_UNBLANK; + bldev->props.brightness = 255; + + minnow_panel_bl_update_status(bldev); + } + + r = omap_dsi_request_vc(dssdev, &mpd->channel); + if (r) { + dev_err(&dssdev->dev, "failed to get virtual channel\n"); + goto err_req_vc; + } + + r = omap_dsi_set_vc_id(dssdev, mpd->channel, TCH); + if (r) { + dev_err(&dssdev->dev, "failed to set VC_ID\n"); + goto err_vc_id; + } + + r = sysfs_create_group(&dssdev->dev.kobj, &minnow_panel_attr_group); + if (r) { + dev_err(&dssdev->dev, "failed to create sysfs files\n"); + goto err_vc_id; + } + + return 0; + +err_vc_id: + omap_dsi_release_vc(dssdev, mpd->channel); +err_req_vc: + if (bldev != NULL) + backlight_device_unregister(bldev); +err_bl: + destroy_workqueue(mpd->workqueue); + return r; +} + +static void __exit minnow_panel_remove(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + struct backlight_device *bldev; + + dev_dbg(&dssdev->dev, "remove\n"); + + sysfs_remove_group(&dssdev->dev.kobj, &minnow_panel_attr_group); + omap_dsi_release_vc(dssdev, mpd->channel); + + bldev = mpd->bldev; + if (bldev != NULL) { + bldev->props.power = FB_BLANK_POWERDOWN; + minnow_panel_bl_update_status(bldev); + backlight_device_unregister(bldev); + } + + minnow_panel_cancel_ulps_work(dssdev); + minnow_panel_cancel_esd_work(dssdev); + destroy_workqueue(mpd->workqueue); + + /* reset, to be sure that the panel is in a valid state */ + minnow_panel_hw_reset(dssdev); +} + +static int minnow_panel_power_on(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + u8 id1, id2, id3; + int r; + + r = omapdss_dsi_configure_pins(dssdev, &mpd->pin_config); + if (r) { + dev_err(&dssdev->dev, "failed to configure DSI pins\n"); + goto err0; + }; + + mpd->dsi_config.timings = &dssdev->panel.timings; + r = omapdss_dsi_set_config(dssdev, &mpd->dsi_config); + if (r) { + dev_err(&dssdev->dev, "failed to configure DSI\n"); + goto err0; + } + + r = omapdss_dsi_display_enable(dssdev); + if (r) { + dev_err(&dssdev->dev, "failed to enable DSI\n"); + goto err0; + } + + if (mpd->intro_printed) + minnow_panel_hw_reset(dssdev); + + omapdss_dsi_vc_enable_hs(dssdev, mpd->channel, false); + + r = minnow_panel_sleep_out(mpd); + if (r) + goto err; + + _minnow_panel_init(mpd); + + r = minnow_panel_get_id(mpd, &id1, &id2, &id3); + if (r) + goto err; + + r = minnow_panel_dcs_write_0(mpd, MIPI_DCS_SET_DISPLAY_ON); + if (r) + goto err; + + r = _minnow_panel_enable_te(dssdev, mpd->te_enabled); + if (r) + goto err; + + r = dsi_enable_video_output(dssdev, mpd->channel); + if (r) + goto err; + + mpd->enabled = 1; + + if (!mpd->intro_printed) { + dev_info(&dssdev->dev, "panel revision %02x.%02x.%02x\n", + id1, id2, id3); + mpd->intro_printed = true; + } + + omapdss_dsi_vc_enable_hs(dssdev, mpd->channel, true); + + return 0; +err: + dev_err(&dssdev->dev, "error while enabling panel, issuing HW reset\n"); + + minnow_panel_hw_reset(dssdev); + + omapdss_dsi_display_disable(dssdev, true, false); +err0: + return r; +} + +static void minnow_panel_power_off(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + dsi_disable_video_output(dssdev, mpd->channel); + + r = minnow_panel_dcs_write_0(mpd, MIPI_DCS_SET_DISPLAY_OFF); + if (!r) + r = minnow_panel_sleep_in(mpd); + + if (r) { + dev_err(&dssdev->dev, + "error disabling panel, issuing HW reset\n"); + minnow_panel_hw_reset(dssdev); + } + + omapdss_dsi_display_disable(dssdev, true, false); + + mpd->enabled = 0; +} + +static int minnow_panel_reset(struct omap_dss_device *dssdev) +{ + dev_err(&dssdev->dev, "performing LCD reset\n"); + + minnow_panel_power_off(dssdev); + minnow_panel_hw_reset(dssdev); + return minnow_panel_power_on(dssdev); +} + +static int minnow_panel_enable(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + dev_dbg(&dssdev->dev, "enable\n"); + + mutex_lock(&mpd->lock); + + if (dssdev->state != OMAP_DSS_DISPLAY_DISABLED) { + r = -EINVAL; + goto err; + } + + dsi_bus_lock(dssdev); + + r = minnow_panel_power_on(dssdev); + + dsi_bus_unlock(dssdev); + + if (r) + goto err; + + minnow_panel_queue_esd_work(dssdev); + + dssdev->state = OMAP_DSS_DISPLAY_ACTIVE; + + mutex_unlock(&mpd->lock); + + return 0; +err: + dev_dbg(&dssdev->dev, "enable failed\n"); + mutex_unlock(&mpd->lock); + return r; +} + +static void minnow_panel_disable(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + dev_dbg(&dssdev->dev, "disable\n"); + + mutex_lock(&mpd->lock); + + minnow_panel_cancel_ulps_work(dssdev); + minnow_panel_cancel_esd_work(dssdev); + + dsi_bus_lock(dssdev); + + if (dssdev->state == OMAP_DSS_DISPLAY_ACTIVE) { + int r; + + r = minnow_panel_wake_up(dssdev); + if (!r) + minnow_panel_power_off(dssdev); + } + + dsi_bus_unlock(dssdev); + + dssdev->state = OMAP_DSS_DISPLAY_DISABLED; + + mutex_unlock(&mpd->lock); +} + +static void minnow_panel_framedone_cb(int err, void *data) +{ + struct omap_dss_device *dssdev = data; + dev_dbg(&dssdev->dev, "framedone, err %d\n", err); + dsi_bus_unlock(dssdev); +} + +static irqreturn_t minnow_panel_te_isr(int irq, void *data) +{ + struct omap_dss_device *dssdev = data; + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int old; + int r; + + old = atomic_cmpxchg(&mpd->do_update, 1, 0); + + if (old) { + cancel_delayed_work(&mpd->te_timeout_work); + + r = omap_dsi_update(dssdev, mpd->channel, minnow_panel_framedone_cb, + dssdev); + if (r) + goto err; + } + + return IRQ_HANDLED; +err: + dev_err(&dssdev->dev, "start update failed\n"); + dsi_bus_unlock(dssdev); + return IRQ_HANDLED; +} + +static void minnow_panel_te_timeout_work_callback(struct work_struct *work) +{ + struct minnow_panel_data *mpd = container_of(work, struct minnow_panel_data, + te_timeout_work.work); + struct omap_dss_device *dssdev = mpd->dssdev; + + dev_err(&dssdev->dev, "TE not received for 250ms!\n"); + + atomic_set(&mpd->do_update, 0); + dsi_bus_unlock(dssdev); +} + +static int minnow_panel_update(struct omap_dss_device *dssdev, + u16 x, u16 y, u16 w, u16 h) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + dev_dbg(&dssdev->dev, "update %d, %d, %d x %d\n", x, y, w, h); + + mutex_lock(&mpd->lock); + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (r) + goto err; + + if (!mpd->enabled) { + r = 0; + goto err; + } + + /* XXX no need to send this every frame, but dsi break if not done */ + r = minnow_panel_set_update_window(mpd, 0, 0, + dssdev->panel.timings.x_res, + dssdev->panel.timings.y_res); + if (r) + goto err; + + if (mpd->te_enabled && gpio_is_valid(mpd->ext_te_gpio)) { + schedule_delayed_work(&mpd->te_timeout_work, + msecs_to_jiffies(250)); + atomic_set(&mpd->do_update, 1); + } else { + r = omap_dsi_update(dssdev, mpd->channel, minnow_panel_framedone_cb, + dssdev); + if (r) + goto err; + } + + /* note: no bus_unlock here. unlock is in framedone_cb */ + mutex_unlock(&mpd->lock); + return 0; +err: + dsi_bus_unlock(dssdev); + mutex_unlock(&mpd->lock); + return r; +} + +static int minnow_panel_sync(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + dev_dbg(&dssdev->dev, "sync\n"); + + mutex_lock(&mpd->lock); + dsi_bus_lock(dssdev); + dsi_bus_unlock(dssdev); + mutex_unlock(&mpd->lock); + + dev_dbg(&dssdev->dev, "sync done\n"); + + return 0; +} + +static int _minnow_panel_enable_te(struct omap_dss_device *dssdev, bool enable) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + if (enable) + r = minnow_panel_dcs_write_1(mpd, MIPI_DCS_SET_TEAR_ON, 0); + else + r = minnow_panel_dcs_write_0(mpd, MIPI_DCS_SET_TEAR_OFF); + + if (!gpio_is_valid(mpd->ext_te_gpio)) + omapdss_dsi_enable_te(dssdev, enable); + + /* possible panel bug */ + msleep(100); + + return r; +} + +static int minnow_panel_enable_te(struct omap_dss_device *dssdev, bool enable) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + mutex_lock(&mpd->lock); + + if (mpd->te_enabled == enable) + goto end; + + dsi_bus_lock(dssdev); + + if (mpd->enabled) { + r = minnow_panel_wake_up(dssdev); + if (r) + goto err; + + r = _minnow_panel_enable_te(dssdev, enable); + if (r) + goto err; + } + + mpd->te_enabled = enable; + + dsi_bus_unlock(dssdev); +end: + mutex_unlock(&mpd->lock); + + return 0; +err: + dsi_bus_unlock(dssdev); + mutex_unlock(&mpd->lock); + + return r; +} + +static int minnow_panel_get_te(struct omap_dss_device *dssdev) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + int r; + + mutex_lock(&mpd->lock); + r = mpd->te_enabled; + mutex_unlock(&mpd->lock); + + return r; +} + +static int minnow_panel_run_test(struct omap_dss_device *dssdev, int test_num) +{ + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + u8 id1, id2, id3; + int r; + + mutex_lock(&mpd->lock); + + if (!mpd->enabled) { + r = -ENODEV; + goto err1; + } + + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (r) + goto err2; + + r = minnow_panel_dcs_read_1(mpd, DCS_GET_ID1, &id1); + if (r) + goto err2; + r = minnow_panel_dcs_read_1(mpd, DCS_GET_ID2, &id2); + if (r) + goto err2; + r = minnow_panel_dcs_read_1(mpd, DCS_GET_ID3, &id3); + if (r) + goto err2; + + dsi_bus_unlock(dssdev); + mutex_unlock(&mpd->lock); + return 0; +err2: + dsi_bus_unlock(dssdev); +err1: + mutex_unlock(&mpd->lock); + return r; +} + +static int minnow_panel_memory_read(struct omap_dss_device *dssdev, + void *buf, size_t size, + u16 x, u16 y, u16 w, u16 h) +{ + int r; + int first = 1; + int plen; + unsigned buf_used = 0; + struct minnow_panel_data *mpd = dev_get_drvdata(&dssdev->dev); + + if (size < w * h * 3) + return -ENOMEM; + + mutex_lock(&mpd->lock); + + if (!mpd->enabled) { + r = -ENODEV; + goto err1; + } + + size = min(w * h * 3, + dssdev->panel.timings.x_res * + dssdev->panel.timings.y_res * 3); + + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (r) + goto err2; + + /* plen 1 or 2 goes into short packet. until checksum error is fixed, + * use short packets. plen 32 works, but bigger packets seem to cause + * an error. */ + if (size % 2) + plen = 1; + else + plen = 2; + + minnow_panel_set_update_window(mpd, x, y, w, h); + + r = dsi_vc_set_max_rx_packet_size(dssdev, mpd->channel, plen); + if (r) + goto err2; + + while (buf_used < size) { + u8 dcs_cmd = first ? 0x2e : 0x3e; + first = 0; + + r = dsi_vc_dcs_read(dssdev, mpd->channel, dcs_cmd, + buf + buf_used, size - buf_used); + + if (r < 0) { + dev_err(&dssdev->dev, "read error\n"); + goto err3; + } + + buf_used += r; + + if (r < plen) { + dev_err(&dssdev->dev, "short read\n"); + break; + } + + if (signal_pending(current)) { + dev_err(&dssdev->dev, "signal pending, " + "aborting memory read\n"); + r = -ERESTARTSYS; + goto err3; + } + } + + r = buf_used; + +err3: + dsi_vc_set_max_rx_packet_size(dssdev, mpd->channel, 1); +err2: + dsi_bus_unlock(dssdev); +err1: + mutex_unlock(&mpd->lock); + return r; +} + +static void minnow_panel_ulps_work(struct work_struct *work) +{ + struct minnow_panel_data *mpd = container_of(work, struct minnow_panel_data, + ulps_work.work); + struct omap_dss_device *dssdev = mpd->dssdev; + + mutex_lock(&mpd->lock); + + if (dssdev->state != OMAP_DSS_DISPLAY_ACTIVE || !mpd->enabled) { + mutex_unlock(&mpd->lock); + return; + } + + dsi_bus_lock(dssdev); + + minnow_panel_enter_ulps(dssdev); + + dsi_bus_unlock(dssdev); + mutex_unlock(&mpd->lock); +} + +static void minnow_panel_esd_work(struct work_struct *work) +{ + struct minnow_panel_data *mpd = container_of(work, struct minnow_panel_data, + esd_work.work); + struct omap_dss_device *dssdev = mpd->dssdev; + u8 state1, state2; + int r; + + mutex_lock(&mpd->lock); + + if (!mpd->enabled) { + mutex_unlock(&mpd->lock); + return; + } + + dsi_bus_lock(dssdev); + + r = minnow_panel_wake_up(dssdev); + if (r) { + dev_err(&dssdev->dev, "failed to exit ULPS\n"); + goto err; + } + + r = minnow_panel_dcs_read_1(mpd, MIPI_DCS_GET_DIAGNOSTIC_RESULT, &state1); + if (r) { + dev_err(&dssdev->dev, "failed to read minnow-panel status\n"); + goto err; + } + + /* Run self diagnostics */ + r = minnow_panel_sleep_out(mpd); + if (r) { + dev_err(&dssdev->dev, "failed to run minnow-panel self-diagnostics\n"); + goto err; + } + + r = minnow_panel_dcs_read_1(mpd, MIPI_DCS_GET_DIAGNOSTIC_RESULT, &state2); + if (r) { + dev_err(&dssdev->dev, "failed to read minnow-panel status\n"); + goto err; + } + + /* Each sleep out command will trigger a self diagnostic and flip + * Bit6 if the test passes. + */ + if (!((state1 ^ state2) & (1 << 6))) { + dev_err(&dssdev->dev, "LCD self diagnostics failed\n"); + goto err; + } + /* Self-diagnostics result is also shown on TE GPIO line. We need + * to re-enable TE after self diagnostics */ + if (mpd->te_enabled && gpio_is_valid(mpd->ext_te_gpio)) { + r = minnow_panel_dcs_write_1(mpd, MIPI_DCS_SET_TEAR_ON, 0); + if (r) + goto err; + } + + dsi_bus_unlock(dssdev); + + minnow_panel_queue_esd_work(dssdev); + + mutex_unlock(&mpd->lock); + + return; +err: + dev_err(&dssdev->dev, "performing LCD reset\n"); + + minnow_panel_reset(dssdev); + + dsi_bus_unlock(dssdev); + + minnow_panel_queue_esd_work(dssdev); + + mutex_unlock(&mpd->lock); +} + +static struct omap_dss_driver minnow_panel_driver = { + .probe = minnow_panel_probe, + .remove = __exit_p(minnow_panel_remove), + + .enable = minnow_panel_enable, + .disable = minnow_panel_disable, + + .update = minnow_panel_update, + .sync = minnow_panel_sync, + + .get_resolution = minnow_panel_get_resolution, + .get_recommended_bpp = omapdss_default_get_recommended_bpp, + + .enable_te = minnow_panel_enable_te, + .get_te = minnow_panel_get_te, + + .run_test = minnow_panel_run_test, + .memory_read = minnow_panel_memory_read, + + .driver = { + .name = "minnow-panel", + .owner = THIS_MODULE, + }, +}; + +static int __init minnow_panel_init(void) +{ + omap_dss_register_driver(&minnow_panel_driver); + + return 0; +} + +static void __exit minnow_panel_exit(void) +{ + omap_dss_unregister_driver(&minnow_panel_driver); +} + +module_init(minnow_panel_init); +module_exit(minnow_panel_exit); + +MODULE_DESCRIPTION("Minnow Panel DSI Driver"); +MODULE_LICENSE("GPL"); |