panel-taal.c 38.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31
/*
 * Taal DSI command mode panel
 *
 * Copyright (C) 2009 Nokia Corporation
 * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com>
 *
 * 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/workqueue.h>
32
#include <linux/slab.h>
33
#include <linux/regulator/consumer.h>
34
#include <linux/mutex.h>
35

36
#include <video/omapdss.h>
37
#include <video/omap-panel-nokia-dsi.h>
38
#include <video/mipi_display.h>
39 40 41 42 43 44 45 46 47 48 49 50 51

/* 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

52 53
static irqreturn_t taal_te_isr(int irq, void *data);
static void taal_te_timeout_work_callback(struct work_struct *work);
54 55
static int _taal_enable_te(struct omap_dss_device *dssdev, bool enable);

56 57
static int taal_panel_reset(struct omap_dss_device *dssdev);

58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
struct panel_regulator {
	struct regulator *regulator;
	const char *name;
	int min_uV;
	int max_uV;
};

static void free_regulators(struct panel_regulator *regulators, int n)
{
	int i;

	for (i = 0; i < n; i++) {
		/* disable/put in reverse order */
		regulator_disable(regulators[n - i - 1].regulator);
		regulator_put(regulators[n - i - 1].regulator);
	}
}

static int init_regulators(struct omap_dss_device *dssdev,
			struct panel_regulator *regulators, int n)
{
	int r, i, v;

	for (i = 0; i < n; i++) {
		struct regulator *reg;

		reg = regulator_get(&dssdev->dev, regulators[i].name);
		if (IS_ERR(reg)) {
			dev_err(&dssdev->dev, "failed to get regulator %s\n",
				regulators[i].name);
			r = PTR_ERR(reg);
			goto err;
		}

		/* FIXME: better handling of fixed vs. variable regulators */
		v = regulator_get_voltage(reg);
		if (v < regulators[i].min_uV || v > regulators[i].max_uV) {
			r = regulator_set_voltage(reg, regulators[i].min_uV,
						regulators[i].max_uV);
			if (r) {
				dev_err(&dssdev->dev,
					"failed to set regulator %s voltage\n",
					regulators[i].name);
				regulator_put(reg);
				goto err;
			}
		}

		r = regulator_enable(reg);
		if (r) {
			dev_err(&dssdev->dev, "failed to enable regulator %s\n",
				regulators[i].name);
			regulator_put(reg);
			goto err;
		}

		regulators[i].regulator = reg;
	}

	return 0;

err:
	free_regulators(regulators, i);

	return r;
}

125 126 127 128 129 130 131
/**
 * struct panel_config - panel configuration
 * @name: panel name
 * @type: panel type
 * @timings: panel resolution
 * @sleep: various panel specific delays, passed to msleep() if non-zero
 * @reset_sequence: reset sequence timings, passed to udelay() if non-zero
132 133
 * @regulators: array of panel regulators
 * @num_regulators: number of regulators in the array
134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151
 */
struct panel_config {
	const char *name;
	int type;

	struct omap_video_timings timings;

	struct {
		unsigned int sleep_in;
		unsigned int sleep_out;
		unsigned int hw_reset;
		unsigned int enable_te;
	} sleep;

	struct {
		unsigned int high;
		unsigned int low;
	} reset_sequence;
152 153 154

	struct panel_regulator *regulators;
	int num_regulators;
155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181
};

enum {
	PANEL_TAAL,
};

static struct panel_config panel_configs[] = {
	{
		.name		= "taal",
		.type		= PANEL_TAAL,
		.timings	= {
			.x_res		= 864,
			.y_res		= 480,
		},
		.sleep		= {
			.sleep_in	= 5,
			.sleep_out	= 5,
			.hw_reset	= 5,
			.enable_te	= 100, /* possible panel bug */
		},
		.reset_sequence	= {
			.high		= 10,
			.low		= 10,
		},
	},
};

182
struct taal_data {
183 184
	struct mutex lock;

185 186 187 188 189 190 191 192 193 194 195 196 197 198
	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;

	bool enabled;
	u8 rotate;
	bool mirror;

	bool te_enabled;
199 200 201 202 203 204 205 206

	atomic_t do_update;
	struct {
		u16 x;
		u16 y;
		u16 w;
		u16 h;
	} update_region;
207 208
	int channel;

209
	struct delayed_work te_timeout_work;
210 211 212 213 214 215 216 217

	bool use_dsi_bl;

	bool cabc_broken;
	unsigned cabc_mode;

	bool intro_printed;

218 219
	struct workqueue_struct *workqueue;

220
	struct delayed_work esd_work;
221
	unsigned esd_interval;
222

223 224 225 226
	bool ulps_enabled;
	unsigned ulps_timeout;
	struct delayed_work ulps_work;

227
	struct panel_config *panel_config;
228 229
};

230 231 232 233 234 235
static inline struct nokia_dsi_panel_data
*get_panel_data(const struct omap_dss_device *dssdev)
{
	return (struct nokia_dsi_panel_data *) dssdev->data;
}

236
static void taal_esd_work(struct work_struct *work);
237
static void taal_ulps_work(struct work_struct *work);
238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254

static void hw_guard_start(struct taal_data *td, int guard_msec)
{
	td->hw_guard_wait = msecs_to_jiffies(guard_msec);
	td->hw_guard_end = jiffies + td->hw_guard_wait;
}

static void hw_guard_wait(struct taal_data *td)
{
	unsigned long wait = td->hw_guard_end - jiffies;

	if ((long)wait > 0 && wait <= td->hw_guard_wait) {
		set_current_state(TASK_UNINTERRUPTIBLE);
		schedule_timeout(wait);
	}
}

255
static int taal_dcs_read_1(struct taal_data *td, u8 dcs_cmd, u8 *data)
256 257 258 259
{
	int r;
	u8 buf[1];

260
	r = dsi_vc_dcs_read(td->dssdev, td->channel, dcs_cmd, buf, 1);
261 262 263 264 265 266 267 268 269

	if (r < 0)
		return r;

	*data = buf[0];

	return 0;
}

270
static int taal_dcs_write_0(struct taal_data *td, u8 dcs_cmd)
271
{
272
	return dsi_vc_dcs_write(td->dssdev, td->channel, &dcs_cmd, 1);
273 274
}

275
static int taal_dcs_write_1(struct taal_data *td, u8 dcs_cmd, u8 param)
276 277 278 279
{
	u8 buf[2];
	buf[0] = dcs_cmd;
	buf[1] = param;
280
	return dsi_vc_dcs_write(td->dssdev, td->channel, buf, 2);
281 282 283 284 285 286 287 288 289 290
}

static int taal_sleep_in(struct taal_data *td)

{
	u8 cmd;
	int r;

	hw_guard_wait(td);

291
	cmd = MIPI_DCS_ENTER_SLEEP_MODE;
292
	r = dsi_vc_dcs_write_nosync(td->dssdev, td->channel, &cmd, 1);
293 294 295 296 297
	if (r)
		return r;

	hw_guard_start(td, 120);

298 299
	if (td->panel_config->sleep.sleep_in)
		msleep(td->panel_config->sleep.sleep_in);
300 301 302 303 304 305 306 307 308 309

	return 0;
}

static int taal_sleep_out(struct taal_data *td)
{
	int r;

	hw_guard_wait(td);

310
	r = taal_dcs_write_0(td, MIPI_DCS_EXIT_SLEEP_MODE);
311 312 313 314 315
	if (r)
		return r;

	hw_guard_start(td, 120);

316 317
	if (td->panel_config->sleep.sleep_out)
		msleep(td->panel_config->sleep.sleep_out);
318 319 320 321

	return 0;
}

322
static int taal_get_id(struct taal_data *td, u8 *id1, u8 *id2, u8 *id3)
323 324 325
{
	int r;

326
	r = taal_dcs_read_1(td, DCS_GET_ID1, id1);
327 328
	if (r)
		return r;
329
	r = taal_dcs_read_1(td, DCS_GET_ID2, id2);
330 331
	if (r)
		return r;
332
	r = taal_dcs_read_1(td, DCS_GET_ID3, id3);
333 334 335 336 337 338
	if (r)
		return r;

	return 0;
}

339
static int taal_set_addr_mode(struct taal_data *td, u8 rotate, bool mirror)
340 341 342 343 344
{
	int r;
	u8 mode;
	int b5, b6, b7;

345
	r = taal_dcs_read_1(td, MIPI_DCS_GET_ADDRESS_MODE, &mode);
346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378
	if (r)
		return r;

	switch (rotate) {
	default:
	case 0:
		b7 = 0;
		b6 = 0;
		b5 = 0;
		break;
	case 1:
		b7 = 0;
		b6 = 1;
		b5 = 1;
		break;
	case 2:
		b7 = 1;
		b6 = 1;
		b5 = 0;
		break;
	case 3:
		b7 = 1;
		b6 = 0;
		b5 = 1;
		break;
	}

	if (mirror)
		b6 = !b6;

	mode &= ~((1<<7) | (1<<6) | (1<<5));
	mode |= (b7 << 7) | (b6 << 6) | (b5 << 5);

379
	return taal_dcs_write_1(td, MIPI_DCS_SET_ADDRESS_MODE, mode);
380 381
}

382 383
static int taal_set_update_window(struct taal_data *td,
		u16 x, u16 y, u16 w, u16 h)
384 385 386 387 388 389 390 391
{
	int r;
	u16 x1 = x;
	u16 x2 = x + w - 1;
	u16 y1 = y;
	u16 y2 = y + h - 1;

	u8 buf[5];
392
	buf[0] = MIPI_DCS_SET_COLUMN_ADDRESS;
393 394 395 396 397
	buf[1] = (x1 >> 8) & 0xff;
	buf[2] = (x1 >> 0) & 0xff;
	buf[3] = (x2 >> 8) & 0xff;
	buf[4] = (x2 >> 0) & 0xff;

398
	r = dsi_vc_dcs_write_nosync(td->dssdev, td->channel, buf, sizeof(buf));
399 400 401
	if (r)
		return r;

402
	buf[0] = MIPI_DCS_SET_PAGE_ADDRESS;
403 404 405 406 407
	buf[1] = (y1 >> 8) & 0xff;
	buf[2] = (y1 >> 0) & 0xff;
	buf[3] = (y2 >> 8) & 0xff;
	buf[4] = (y2 >> 0) & 0xff;

408
	r = dsi_vc_dcs_write_nosync(td->dssdev, td->channel, buf, sizeof(buf));
409 410 411
	if (r)
		return r;

412
	dsi_vc_send_bta_sync(td->dssdev, td->channel);
413 414 415 416

	return r;
}

417 418 419 420 421
static void taal_queue_esd_work(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

	if (td->esd_interval > 0)
422
		queue_delayed_work(td->workqueue, &td->esd_work,
423 424 425 426 427 428 429 430 431 432
				msecs_to_jiffies(td->esd_interval));
}

static void taal_cancel_esd_work(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

	cancel_delayed_work(&td->esd_work);
}

433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492
static void taal_queue_ulps_work(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

	if (td->ulps_timeout > 0)
		queue_delayed_work(td->workqueue, &td->ulps_work,
				msecs_to_jiffies(td->ulps_timeout));
}

static void taal_cancel_ulps_work(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

	cancel_delayed_work(&td->ulps_work);
}

static int taal_enter_ulps(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
	int r;

	if (td->ulps_enabled)
		return 0;

	taal_cancel_ulps_work(dssdev);

	r = _taal_enable_te(dssdev, false);
	if (r)
		goto err;

	disable_irq(gpio_to_irq(panel_data->ext_te_gpio));

	omapdss_dsi_display_disable(dssdev, false, true);

	td->ulps_enabled = true;

	return 0;

err:
	dev_err(&dssdev->dev, "enter ULPS failed");
	taal_panel_reset(dssdev);

	td->ulps_enabled = false;

	taal_queue_ulps_work(dssdev);

	return r;
}

static int taal_exit_ulps(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
	int r;

	if (!td->ulps_enabled)
		return 0;

	r = omapdss_dsi_display_enable(dssdev);
493 494 495 496
	if (r) {
		dev_err(&dssdev->dev, "failed to enable DSI\n");
		goto err1;
	}
497

498
	omapdss_dsi_vc_enable_hs(dssdev, td->channel, true);
499 500

	r = _taal_enable_te(dssdev, true);
501 502 503 504
	if (r) {
		dev_err(&dssdev->dev, "failed to re-enable TE");
		goto err2;
	}
505 506 507 508 509 510 511 512 513

	enable_irq(gpio_to_irq(panel_data->ext_te_gpio));

	taal_queue_ulps_work(dssdev);

	td->ulps_enabled = false;

	return 0;

514 515
err2:
	dev_err(&dssdev->dev, "failed to exit ULPS");
516

517 518 519 520 521 522
	r = taal_panel_reset(dssdev);
	if (!r) {
		enable_irq(gpio_to_irq(panel_data->ext_te_gpio));
		td->ulps_enabled = false;
	}
err1:
523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539
	taal_queue_ulps_work(dssdev);

	return r;
}

static int taal_wake_up(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

	if (td->ulps_enabled)
		return taal_exit_ulps(dssdev);

	taal_cancel_ulps_work(dssdev);
	taal_queue_ulps_work(dssdev);
	return 0;
}

540 541 542 543
static int taal_bl_update_status(struct backlight_device *dev)
{
	struct omap_dss_device *dssdev = dev_get_drvdata(&dev->dev);
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
544
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
545 546 547 548 549 550 551 552 553 554 555
	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);

556 557
	mutex_lock(&td->lock);

558 559
	if (td->use_dsi_bl) {
		if (td->enabled) {
560
			dsi_bus_lock(dssdev);
561 562 563 564 565

			r = taal_wake_up(dssdev);
			if (!r)
				r = taal_dcs_write_1(td, DCS_BRIGHTNESS, level);

566
			dsi_bus_unlock(dssdev);
567 568
		} else {
			r = 0;
569 570
		}
	} else {
571
		if (!panel_data->set_backlight)
572 573
			r = -EINVAL;
		else
574
			r = panel_data->set_backlight(dssdev, level);
575 576
	}

577 578 579
	mutex_unlock(&td->lock);

	return r;
580 581 582 583 584 585 586 587 588 589 590
}

static int taal_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;
}

L
Lionel Debroux 已提交
591
static const struct backlight_ops taal_bl_ops = {
592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623
	.get_brightness = taal_bl_get_intensity,
	.update_status  = taal_bl_update_status,
};

static void taal_get_timings(struct omap_dss_device *dssdev,
			struct omap_video_timings *timings)
{
	*timings = dssdev->panel.timings;
}

static void taal_get_resolution(struct omap_dss_device *dssdev,
		u16 *xres, u16 *yres)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

	if (td->rotate == 0 || td->rotate == 2) {
		*xres = dssdev->panel.timings.x_res;
		*yres = dssdev->panel.timings.y_res;
	} else {
		*yres = dssdev->panel.timings.x_res;
		*xres = dssdev->panel.timings.y_res;
	}
}

static ssize_t taal_num_errors_show(struct device *dev,
		struct device_attribute *attr, char *buf)
{
	struct omap_dss_device *dssdev = to_dss_device(dev);
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	u8 errors;
	int r;

624 625
	mutex_lock(&td->lock);

626
	if (td->enabled) {
627
		dsi_bus_lock(dssdev);
628 629 630 631 632

		r = taal_wake_up(dssdev);
		if (!r)
			r = taal_dcs_read_1(td, DCS_READ_NUM_ERRORS, &errors);

633
		dsi_bus_unlock(dssdev);
634 635 636 637
	} else {
		r = -ENODEV;
	}

638 639
	mutex_unlock(&td->lock);

640 641 642 643 644 645 646 647 648 649 650 651 652 653
	if (r)
		return r;

	return snprintf(buf, PAGE_SIZE, "%d\n", errors);
}

static ssize_t taal_hw_revision_show(struct device *dev,
		struct device_attribute *attr, char *buf)
{
	struct omap_dss_device *dssdev = to_dss_device(dev);
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	u8 id1, id2, id3;
	int r;

654 655
	mutex_lock(&td->lock);

656
	if (td->enabled) {
657
		dsi_bus_lock(dssdev);
658 659 660 661 662

		r = taal_wake_up(dssdev);
		if (!r)
			r = taal_get_id(td, &id1, &id2, &id3);

663
		dsi_bus_unlock(dssdev);
664 665 666 667
	} else {
		r = -ENODEV;
	}

668 669
	mutex_unlock(&td->lock);

670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709
	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 taal_data *td = dev_get_drvdata(&dssdev->dev);
	const char *mode_str;
	int mode;
	int len;

	mode = td->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 taal_data *td = dev_get_drvdata(&dssdev->dev);
	int i;
710
	int r;
711 712 713 714 715 716 717 718 719

	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;

720 721
	mutex_lock(&td->lock);

722
	if (td->enabled) {
723
		dsi_bus_lock(dssdev);
724 725 726 727 728 729 730 731 732 733 734

		if (!td->cabc_broken) {
			r = taal_wake_up(dssdev);
			if (r)
				goto err;

			r = taal_dcs_write_1(td, DCS_WRITE_CABC, i);
			if (r)
				goto err;
		}

735
		dsi_bus_unlock(dssdev);
736 737 738 739
	}

	td->cabc_mode = i;

740 741
	mutex_unlock(&td->lock);

742
	return count;
743
err:
744
	dsi_bus_unlock(dssdev);
745 746
	mutex_unlock(&td->lock);
	return r;
747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764
}

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;
}

765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803
static ssize_t taal_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 taal_data *td = dev_get_drvdata(&dssdev->dev);

	unsigned long t;
	int r;

	r = strict_strtoul(buf, 10, &t);
	if (r)
		return r;

	mutex_lock(&td->lock);
	taal_cancel_esd_work(dssdev);
	td->esd_interval = t;
	if (td->enabled)
		taal_queue_esd_work(dssdev);
	mutex_unlock(&td->lock);

	return count;
}

static ssize_t taal_show_esd_interval(struct device *dev,
		struct device_attribute *attr,
		char *buf)
{
	struct omap_dss_device *dssdev = to_dss_device(dev);
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	unsigned t;

	mutex_lock(&td->lock);
	t = td->esd_interval;
	mutex_unlock(&td->lock);

	return snprintf(buf, PAGE_SIZE, "%u\n", t);
}

804 805 806 807 808 809 810 811 812 813 814 815 816 817 818 819
static ssize_t taal_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 taal_data *td = dev_get_drvdata(&dssdev->dev);
	unsigned long t;
	int r;

	r = strict_strtoul(buf, 10, &t);
	if (r)
		return r;

	mutex_lock(&td->lock);

	if (td->enabled) {
820
		dsi_bus_lock(dssdev);
821 822 823 824 825 826

		if (t)
			r = taal_enter_ulps(dssdev);
		else
			r = taal_wake_up(dssdev);

827
		dsi_bus_unlock(dssdev);
828 829 830 831 832 833 834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860 861 862 863 864 865 866 867 868 869 870
	}

	mutex_unlock(&td->lock);

	if (r)
		return r;

	return count;
}

static ssize_t taal_show_ulps(struct device *dev,
		struct device_attribute *attr,
		char *buf)
{
	struct omap_dss_device *dssdev = to_dss_device(dev);
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	unsigned t;

	mutex_lock(&td->lock);
	t = td->ulps_enabled;
	mutex_unlock(&td->lock);

	return snprintf(buf, PAGE_SIZE, "%u\n", t);
}

static ssize_t taal_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 taal_data *td = dev_get_drvdata(&dssdev->dev);
	unsigned long t;
	int r;

	r = strict_strtoul(buf, 10, &t);
	if (r)
		return r;

	mutex_lock(&td->lock);
	td->ulps_timeout = t;

	if (td->enabled) {
		/* taal_wake_up will restart the timer */
871
		dsi_bus_lock(dssdev);
872
		r = taal_wake_up(dssdev);
873
		dsi_bus_unlock(dssdev);
874 875 876 877 878 879 880 881 882 883 884 885 886 887 888 889 890 891 892 893 894 895 896 897 898
	}

	mutex_unlock(&td->lock);

	if (r)
		return r;

	return count;
}

static ssize_t taal_show_ulps_timeout(struct device *dev,
		struct device_attribute *attr,
		char *buf)
{
	struct omap_dss_device *dssdev = to_dss_device(dev);
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	unsigned t;

	mutex_lock(&td->lock);
	t = td->ulps_timeout;
	mutex_unlock(&td->lock);

	return snprintf(buf, PAGE_SIZE, "%u\n", t);
}

899 900 901 902 903 904
static DEVICE_ATTR(num_dsi_errors, S_IRUGO, taal_num_errors_show, NULL);
static DEVICE_ATTR(hw_revision, S_IRUGO, taal_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);
905 906
static DEVICE_ATTR(esd_interval, S_IRUGO | S_IWUSR,
		taal_show_esd_interval, taal_store_esd_interval);
907 908 909 910
static DEVICE_ATTR(ulps, S_IRUGO | S_IWUSR,
		taal_show_ulps, taal_store_ulps);
static DEVICE_ATTR(ulps_timeout, S_IRUGO | S_IWUSR,
		taal_show_ulps_timeout, taal_store_ulps_timeout);
911 912 913 914 915 916

static struct attribute *taal_attrs[] = {
	&dev_attr_num_dsi_errors.attr,
	&dev_attr_hw_revision.attr,
	&dev_attr_cabc_mode.attr,
	&dev_attr_cabc_available_modes.attr,
917
	&dev_attr_esd_interval.attr,
918 919
	&dev_attr_ulps.attr,
	&dev_attr_ulps_timeout.attr,
920 921 922 923 924 925 926
	NULL,
};

static struct attribute_group taal_attr_group = {
	.attrs = taal_attrs,
};

927 928
static void taal_hw_reset(struct omap_dss_device *dssdev)
{
929
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
930 931 932
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);

	if (panel_data->reset_gpio == -1)
933 934
		return;

935
	gpio_set_value(panel_data->reset_gpio, 1);
936 937
	if (td->panel_config->reset_sequence.high)
		udelay(td->panel_config->reset_sequence.high);
938
	/* reset the panel */
939
	gpio_set_value(panel_data->reset_gpio, 0);
940 941 942
	/* assert reset */
	if (td->panel_config->reset_sequence.low)
		udelay(td->panel_config->reset_sequence.low);
943
	gpio_set_value(panel_data->reset_gpio, 1);
944 945 946
	/* wait after releasing reset */
	if (td->panel_config->sleep.hw_reset)
		msleep(td->panel_config->sleep.hw_reset);
947 948
}

949 950
static int taal_probe(struct omap_dss_device *dssdev)
{
951
	struct backlight_properties props;
952 953
	struct taal_data *td;
	struct backlight_device *bldev;
954
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
955 956
	struct panel_config *panel_config = NULL;
	int r, i;
957 958 959

	dev_dbg(&dssdev->dev, "probe\n");

960 961 962 963 964
	if (!panel_data || !panel_data->name) {
		r = -EINVAL;
		goto err;
	}

965 966 967 968 969 970 971 972 973 974 975 976
	for (i = 0; i < ARRAY_SIZE(panel_configs); i++) {
		if (strcmp(panel_data->name, panel_configs[i].name) == 0) {
			panel_config = &panel_configs[i];
			break;
		}
	}

	if (!panel_config) {
		r = -EINVAL;
		goto err;
	}

977
	dssdev->panel.config = OMAP_DSS_LCD_TFT;
978
	dssdev->panel.timings = panel_config->timings;
979
	dssdev->panel.dsi_pix_fmt = OMAP_DSS_DSI_FMT_RGB888;
980 981 982 983

	td = kzalloc(sizeof(*td), GFP_KERNEL);
	if (!td) {
		r = -ENOMEM;
984
		goto err;
985 986
	}
	td->dssdev = dssdev;
987
	td->panel_config = panel_config;
988
	td->esd_interval = panel_data->esd_interval;
989 990
	td->ulps_enabled = false;
	td->ulps_timeout = panel_data->ulps_timeout;
991

992 993
	mutex_init(&td->lock);

994 995
	atomic_set(&td->do_update, 0);

996 997 998 999 1000
	r = init_regulators(dssdev, panel_config->regulators,
			panel_config->num_regulators);
	if (r)
		goto err_reg;

1001 1002
	td->workqueue = create_singlethread_workqueue("taal_esd");
	if (td->workqueue == NULL) {
1003 1004
		dev_err(&dssdev->dev, "can't create ESD workqueue\n");
		r = -ENOMEM;
1005
		goto err_wq;
1006 1007
	}
	INIT_DELAYED_WORK_DEFERRABLE(&td->esd_work, taal_esd_work);
1008
	INIT_DELAYED_WORK(&td->ulps_work, taal_ulps_work);
1009 1010 1011

	dev_set_drvdata(&dssdev->dev, td);

1012 1013
	taal_hw_reset(dssdev);

1014 1015
	/* if no platform set_backlight() defined, presume DSI backlight
	 * control */
1016
	memset(&props, 0, sizeof(struct backlight_properties));
1017
	if (!panel_data->set_backlight)
1018 1019
		td->use_dsi_bl = true;

1020 1021 1022 1023
	if (td->use_dsi_bl)
		props.max_brightness = 255;
	else
		props.max_brightness = 127;
M
Matthew Garrett 已提交
1024 1025

	props.type = BACKLIGHT_RAW;
1026 1027
	bldev = backlight_device_register(dev_name(&dssdev->dev), &dssdev->dev,
					dssdev, &taal_bl_ops, &props);
1028 1029
	if (IS_ERR(bldev)) {
		r = PTR_ERR(bldev);
1030
		goto err_bl;
1031 1032 1033 1034 1035 1036
	}

	td->bldev = bldev;

	bldev->props.fb_blank = FB_BLANK_UNBLANK;
	bldev->props.power = FB_BLANK_UNBLANK;
1037
	if (td->use_dsi_bl)
1038
		bldev->props.brightness = 255;
1039
	else
1040 1041 1042 1043
		bldev->props.brightness = 127;

	taal_bl_update_status(bldev);

1044 1045
	if (panel_data->use_ext_te) {
		int gpio = panel_data->ext_te_gpio;
1046 1047 1048 1049

		r = gpio_request(gpio, "taal irq");
		if (r) {
			dev_err(&dssdev->dev, "GPIO request failed\n");
1050
			goto err_gpio;
1051 1052 1053 1054 1055 1056 1057 1058 1059 1060 1061
		}

		gpio_direction_input(gpio);

		r = request_irq(gpio_to_irq(gpio), taal_te_isr,
				IRQF_DISABLED | IRQF_TRIGGER_RISING,
				"taal vsync", dssdev);

		if (r) {
			dev_err(&dssdev->dev, "IRQ request failed\n");
			gpio_free(gpio);
1062
			goto err_irq;
1063 1064
		}

1065 1066
		INIT_DELAYED_WORK_DEFERRABLE(&td->te_timeout_work,
					taal_te_timeout_work_callback);
1067

1068
		dev_dbg(&dssdev->dev, "Using GPIO TE\n");
1069 1070
	}

1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082
	r = omap_dsi_request_vc(dssdev, &td->channel);
	if (r) {
		dev_err(&dssdev->dev, "failed to get virtual channel\n");
		goto err_req_vc;
	}

	r = omap_dsi_set_vc_id(dssdev, td->channel, TCH);
	if (r) {
		dev_err(&dssdev->dev, "failed to set VC_ID\n");
		goto err_vc_id;
	}

1083 1084 1085
	r = sysfs_create_group(&dssdev->dev.kobj, &taal_attr_group);
	if (r) {
		dev_err(&dssdev->dev, "failed to create sysfs files\n");
1086
		goto err_vc_id;
1087 1088 1089
	}

	return 0;
1090 1091 1092 1093

err_vc_id:
	omap_dsi_release_vc(dssdev, td->channel);
err_req_vc:
1094 1095
	if (panel_data->use_ext_te)
		free_irq(gpio_to_irq(panel_data->ext_te_gpio), dssdev);
1096
err_irq:
1097 1098
	if (panel_data->use_ext_te)
		gpio_free(panel_data->ext_te_gpio);
1099
err_gpio:
1100
	backlight_device_unregister(bldev);
1101
err_bl:
1102
	destroy_workqueue(td->workqueue);
1103
err_wq:
1104 1105
	free_regulators(panel_config->regulators, panel_config->num_regulators);
err_reg:
1106
	kfree(td);
1107
err:
1108 1109 1110
	return r;
}

1111
static void __exit taal_remove(struct omap_dss_device *dssdev)
1112 1113
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1114
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
1115 1116 1117 1118 1119
	struct backlight_device *bldev;

	dev_dbg(&dssdev->dev, "remove\n");

	sysfs_remove_group(&dssdev->dev.kobj, &taal_attr_group);
1120
	omap_dsi_release_vc(dssdev, td->channel);
1121

1122 1123
	if (panel_data->use_ext_te) {
		int gpio = panel_data->ext_te_gpio;
1124 1125 1126 1127 1128 1129 1130 1131 1132
		free_irq(gpio_to_irq(gpio), dssdev);
		gpio_free(gpio);
	}

	bldev = td->bldev;
	bldev->props.power = FB_BLANK_POWERDOWN;
	taal_bl_update_status(bldev);
	backlight_device_unregister(bldev);

1133
	taal_cancel_ulps_work(dssdev);
1134
	taal_cancel_esd_work(dssdev);
1135
	destroy_workqueue(td->workqueue);
1136

1137 1138 1139
	/* reset, to be sure that the panel is in a valid state */
	taal_hw_reset(dssdev);

1140 1141 1142
	free_regulators(td->panel_config->regulators,
			td->panel_config->num_regulators);

1143 1144 1145
	kfree(td);
}

1146
static int taal_power_on(struct omap_dss_device *dssdev)
1147 1148 1149 1150 1151
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	u8 id1, id2, id3;
	int r;

1152 1153 1154 1155 1156 1157
	r = omapdss_dsi_display_enable(dssdev);
	if (r) {
		dev_err(&dssdev->dev, "failed to enable DSI\n");
		goto err0;
	}

1158 1159
	taal_hw_reset(dssdev);

1160
	omapdss_dsi_vc_enable_hs(dssdev, td->channel, false);
1161

1162 1163 1164 1165
	r = taal_sleep_out(td);
	if (r)
		goto err;

1166
	r = taal_get_id(td, &id1, &id2, &id3);
1167 1168 1169
	if (r)
		goto err;

1170 1171 1172
	/* on early Taal revisions CABC is broken */
	if (td->panel_config->type == PANEL_TAAL &&
		(id2 == 0x00 || id2 == 0xff || id2 == 0x81))
1173 1174
		td->cabc_broken = true;

1175
	r = taal_dcs_write_1(td, DCS_BRIGHTNESS, 0xff);
1176 1177
	if (r)
		goto err;
1178

1179
	r = taal_dcs_write_1(td, DCS_CTRL_DISPLAY,
1180 1181 1182
			(1<<2) | (1<<5));	/* BL | BCTRL */
	if (r)
		goto err;
1183

1184 1185
	r = taal_dcs_write_1(td, MIPI_DCS_SET_PIXEL_FORMAT,
		MIPI_DCS_PIXEL_FMT_24BIT);
1186 1187
	if (r)
		goto err;
1188

1189
	r = taal_set_addr_mode(td, td->rotate, td->mirror);
1190 1191 1192 1193
	if (r)
		goto err;

	if (!td->cabc_broken) {
1194
		r = taal_dcs_write_1(td, DCS_WRITE_CABC, td->cabc_mode);
1195 1196 1197 1198
		if (r)
			goto err;
	}

1199
	r = taal_dcs_write_0(td, MIPI_DCS_SET_DISPLAY_ON);
1200 1201
	if (r)
		goto err;
1202

1203 1204 1205 1206
	r = _taal_enable_te(dssdev, td->te_enabled);
	if (r)
		goto err;

1207 1208 1209
	td->enabled = 1;

	if (!td->intro_printed) {
1210 1211
		dev_info(&dssdev->dev, "%s panel revision %02x.%02x.%02x\n",
			td->panel_config->name, id1, id2, id3);
1212 1213 1214 1215 1216 1217
		if (td->cabc_broken)
			dev_info(&dssdev->dev,
					"old Taal version, CABC disabled\n");
		td->intro_printed = true;
	}

1218
	omapdss_dsi_vc_enable_hs(dssdev, td->channel, true);
1219

1220 1221
	return 0;
err:
1222 1223 1224 1225
	dev_err(&dssdev->dev, "error while enabling panel, issuing HW reset\n");

	taal_hw_reset(dssdev);

1226
	omapdss_dsi_display_disable(dssdev, true, false);
1227
err0:
1228 1229 1230
	return r;
}

1231
static void taal_power_off(struct omap_dss_device *dssdev)
1232 1233
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1234
	int r;
1235

1236
	r = taal_dcs_write_0(td, MIPI_DCS_SET_DISPLAY_OFF);
1237
	if (!r)
1238
		r = taal_sleep_in(td);
1239

1240 1241 1242 1243 1244
	if (r) {
		dev_err(&dssdev->dev,
				"error disabling panel, issuing HW reset\n");
		taal_hw_reset(dssdev);
	}
1245

1246
	omapdss_dsi_display_disable(dssdev, true, false);
1247

1248
	td->enabled = 0;
1249 1250
}

1251 1252 1253 1254 1255 1256 1257 1258 1259
static int taal_panel_reset(struct omap_dss_device *dssdev)
{
	dev_err(&dssdev->dev, "performing LCD reset\n");

	taal_power_off(dssdev);
	taal_hw_reset(dssdev);
	return taal_power_on(dssdev);
}

1260 1261
static int taal_enable(struct omap_dss_device *dssdev)
{
1262
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1263
	int r;
1264

1265 1266
	dev_dbg(&dssdev->dev, "enable\n");

1267 1268 1269 1270 1271 1272
	mutex_lock(&td->lock);

	if (dssdev->state != OMAP_DSS_DISPLAY_DISABLED) {
		r = -EINVAL;
		goto err;
	}
1273

1274
	dsi_bus_lock(dssdev);
1275

1276
	r = taal_power_on(dssdev);
1277

1278
	dsi_bus_unlock(dssdev);
1279

1280
	if (r)
1281
		goto err;
1282

1283
	taal_queue_esd_work(dssdev);
1284

1285 1286
	dssdev->state = OMAP_DSS_DISPLAY_ACTIVE;

1287 1288 1289 1290 1291 1292
	mutex_unlock(&td->lock);

	return 0;
err:
	dev_dbg(&dssdev->dev, "enable failed\n");
	mutex_unlock(&td->lock);
1293 1294 1295 1296 1297
	return r;
}

static void taal_disable(struct omap_dss_device *dssdev)
{
1298 1299
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

1300 1301
	dev_dbg(&dssdev->dev, "disable\n");

1302 1303
	mutex_lock(&td->lock);

1304
	taal_cancel_ulps_work(dssdev);
1305
	taal_cancel_esd_work(dssdev);
1306

1307
	dsi_bus_lock(dssdev);
1308

1309
	if (dssdev->state == OMAP_DSS_DISPLAY_ACTIVE) {
1310 1311 1312 1313 1314
		int r;

		r = taal_wake_up(dssdev);
		if (!r)
			taal_power_off(dssdev);
1315
	}
1316

1317
	dsi_bus_unlock(dssdev);
1318

1319
	dssdev->state = OMAP_DSS_DISPLAY_DISABLED;
1320 1321

	mutex_unlock(&td->lock);
1322 1323 1324 1325
}

static int taal_suspend(struct omap_dss_device *dssdev)
{
1326 1327 1328
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	int r;

1329
	dev_dbg(&dssdev->dev, "suspend\n");
1330

1331 1332 1333 1334 1335 1336
	mutex_lock(&td->lock);

	if (dssdev->state != OMAP_DSS_DISPLAY_ACTIVE) {
		r = -EINVAL;
		goto err;
	}
1337

1338
	taal_cancel_ulps_work(dssdev);
1339
	taal_cancel_esd_work(dssdev);
1340

1341
	dsi_bus_lock(dssdev);
1342

1343 1344 1345
	r = taal_wake_up(dssdev);
	if (!r)
		taal_power_off(dssdev);
1346

1347
	dsi_bus_unlock(dssdev);
1348

1349
	dssdev->state = OMAP_DSS_DISPLAY_SUSPENDED;
1350

1351 1352
	mutex_unlock(&td->lock);

1353
	return 0;
1354 1355 1356
err:
	mutex_unlock(&td->lock);
	return r;
1357 1358 1359 1360
}

static int taal_resume(struct omap_dss_device *dssdev)
{
1361
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1362
	int r;
1363

1364
	dev_dbg(&dssdev->dev, "resume\n");
1365

1366 1367 1368 1369 1370 1371
	mutex_lock(&td->lock);

	if (dssdev->state != OMAP_DSS_DISPLAY_SUSPENDED) {
		r = -EINVAL;
		goto err;
	}
1372

1373
	dsi_bus_lock(dssdev);
1374

1375
	r = taal_power_on(dssdev);
1376

1377
	dsi_bus_unlock(dssdev);
1378

1379
	if (r) {
1380
		dssdev->state = OMAP_DSS_DISPLAY_DISABLED;
1381
	} else {
1382
		dssdev->state = OMAP_DSS_DISPLAY_ACTIVE;
1383
		taal_queue_esd_work(dssdev);
1384
	}
1385 1386 1387 1388 1389 1390

	mutex_unlock(&td->lock);

	return r;
err:
	mutex_unlock(&td->lock);
1391
	return r;
1392 1393
}

1394 1395 1396 1397
static void taal_framedone_cb(int err, void *data)
{
	struct omap_dss_device *dssdev = data;
	dev_dbg(&dssdev->dev, "framedone, err %d\n", err);
1398
	dsi_bus_unlock(dssdev);
1399 1400
}

1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412
static irqreturn_t taal_te_isr(int irq, void *data)
{
	struct omap_dss_device *dssdev = data;
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	int old;
	int r;

	old = atomic_cmpxchg(&td->do_update, 1, 0);

	if (old) {
		cancel_delayed_work(&td->te_timeout_work);

1413
		r = omap_dsi_update(dssdev, td->channel,
1414 1415 1416 1417 1418 1419 1420 1421 1422 1423 1424 1425
				td->update_region.x,
				td->update_region.y,
				td->update_region.w,
				td->update_region.h,
				taal_framedone_cb, dssdev);
		if (r)
			goto err;
	}

	return IRQ_HANDLED;
err:
	dev_err(&dssdev->dev, "start update failed\n");
1426
	dsi_bus_unlock(dssdev);
1427 1428 1429 1430 1431 1432 1433 1434 1435 1436 1437 1438
	return IRQ_HANDLED;
}

static void taal_te_timeout_work_callback(struct work_struct *work)
{
	struct taal_data *td = container_of(work, struct taal_data,
					te_timeout_work.work);
	struct omap_dss_device *dssdev = td->dssdev;

	dev_err(&dssdev->dev, "TE not received for 250ms!\n");

	atomic_set(&td->do_update, 0);
1439
	dsi_bus_unlock(dssdev);
1440 1441
}

1442
static int taal_update(struct omap_dss_device *dssdev,
1443 1444
				    u16 x, u16 y, u16 w, u16 h)
{
1445
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1446
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
1447 1448 1449 1450
	int r;

	dev_dbg(&dssdev->dev, "update %d, %d, %d x %d\n", x, y, w, h);

1451
	mutex_lock(&td->lock);
1452
	dsi_bus_lock(dssdev);
1453

1454 1455 1456 1457
	r = taal_wake_up(dssdev);
	if (r)
		goto err;

1458 1459 1460 1461 1462
	if (!td->enabled) {
		r = 0;
		goto err;
	}

1463
	r = omap_dsi_prepare_update(dssdev, &x, &y, &w, &h, true);
1464 1465 1466
	if (r)
		goto err;

1467
	r = taal_set_update_window(td, x, y, w, h);
1468 1469 1470
	if (r)
		goto err;

1471
	if (td->te_enabled && panel_data->use_ext_te) {
1472 1473 1474 1475 1476 1477 1478 1479 1480
		td->update_region.x = x;
		td->update_region.y = y;
		td->update_region.w = w;
		td->update_region.h = h;
		barrier();
		schedule_delayed_work(&td->te_timeout_work,
				msecs_to_jiffies(250));
		atomic_set(&td->do_update, 1);
	} else {
1481
		r = omap_dsi_update(dssdev, td->channel, x, y, w, h,
1482 1483 1484 1485
				taal_framedone_cb, dssdev);
		if (r)
			goto err;
	}
1486 1487

	/* note: no bus_unlock here. unlock is in framedone_cb */
1488
	mutex_unlock(&td->lock);
1489 1490
	return 0;
err:
1491
	dsi_bus_unlock(dssdev);
1492
	mutex_unlock(&td->lock);
1493 1494 1495 1496 1497
	return r;
}

static int taal_sync(struct omap_dss_device *dssdev)
{
1498 1499
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

1500 1501
	dev_dbg(&dssdev->dev, "sync\n");

1502
	mutex_lock(&td->lock);
1503 1504
	dsi_bus_lock(dssdev);
	dsi_bus_unlock(dssdev);
1505
	mutex_unlock(&td->lock);
1506 1507 1508 1509

	dev_dbg(&dssdev->dev, "sync done\n");

	return 0;
1510 1511
}

1512
static int _taal_enable_te(struct omap_dss_device *dssdev, bool enable)
1513
{
1514
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1515
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
1516 1517 1518
	int r;

	if (enable)
1519
		r = taal_dcs_write_1(td, MIPI_DCS_SET_TEAR_ON, 0);
1520
	else
1521
		r = taal_dcs_write_0(td, MIPI_DCS_SET_TEAR_OFF);
1522

1523
	if (!panel_data->use_ext_te)
1524
		omapdss_dsi_enable_te(dssdev, enable);
1525

1526 1527
	if (td->panel_config->sleep.enable_te)
		msleep(td->panel_config->sleep.enable_te);
1528

1529 1530 1531 1532 1533
	return r;
}

static int taal_enable_te(struct omap_dss_device *dssdev, bool enable)
{
1534
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1535 1536
	int r;

1537
	mutex_lock(&td->lock);
1538 1539 1540 1541

	if (td->te_enabled == enable)
		goto end;

1542
	dsi_bus_lock(dssdev);
1543

1544
	if (td->enabled) {
1545 1546 1547 1548
		r = taal_wake_up(dssdev);
		if (r)
			goto err;

1549 1550 1551 1552 1553 1554
		r = _taal_enable_te(dssdev, enable);
		if (r)
			goto err;
	}

	td->te_enabled = enable;
1555

1556
	dsi_bus_unlock(dssdev);
1557
end:
1558
	mutex_unlock(&td->lock);
1559

1560 1561
	return 0;
err:
1562
	dsi_bus_unlock(dssdev);
1563 1564
	mutex_unlock(&td->lock);

1565 1566 1567
	return r;
}

1568 1569 1570
static int taal_get_te(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1571 1572 1573 1574 1575 1576 1577
	int r;

	mutex_lock(&td->lock);
	r = td->te_enabled;
	mutex_unlock(&td->lock);

	return r;
1578 1579
}

1580 1581 1582 1583 1584 1585 1586
static int taal_rotate(struct omap_dss_device *dssdev, u8 rotate)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	int r;

	dev_dbg(&dssdev->dev, "rotate %d\n", rotate);

1587
	mutex_lock(&td->lock);
1588 1589 1590 1591

	if (td->rotate == rotate)
		goto end;

1592
	dsi_bus_lock(dssdev);
1593

1594
	if (td->enabled) {
1595 1596 1597 1598
		r = taal_wake_up(dssdev);
		if (r)
			goto err;

1599
		r = taal_set_addr_mode(td, rotate, td->mirror);
1600
		if (r)
1601
			goto err;
1602 1603 1604 1605
	}

	td->rotate = rotate;

1606
	dsi_bus_unlock(dssdev);
1607
end:
1608
	mutex_unlock(&td->lock);
1609
	return 0;
1610
err:
1611
	dsi_bus_unlock(dssdev);
1612
	mutex_unlock(&td->lock);
1613
	return r;
1614 1615 1616 1617 1618
}

static u8 taal_get_rotate(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1619 1620 1621 1622 1623 1624 1625
	int r;

	mutex_lock(&td->lock);
	r = td->rotate;
	mutex_unlock(&td->lock);

	return r;
1626 1627 1628 1629 1630 1631 1632 1633 1634
}

static int taal_mirror(struct omap_dss_device *dssdev, bool enable)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
	int r;

	dev_dbg(&dssdev->dev, "mirror %d\n", enable);

1635
	mutex_lock(&td->lock);
1636 1637 1638 1639

	if (td->mirror == enable)
		goto end;

1640
	dsi_bus_lock(dssdev);
1641
	if (td->enabled) {
1642 1643 1644 1645
		r = taal_wake_up(dssdev);
		if (r)
			goto err;

1646
		r = taal_set_addr_mode(td, td->rotate, enable);
1647
		if (r)
1648
			goto err;
1649 1650 1651 1652
	}

	td->mirror = enable;

1653
	dsi_bus_unlock(dssdev);
1654
end:
1655
	mutex_unlock(&td->lock);
1656
	return 0;
1657
err:
1658
	dsi_bus_unlock(dssdev);
1659
	mutex_unlock(&td->lock);
1660
	return r;
1661 1662 1663 1664 1665
}

static bool taal_get_mirror(struct omap_dss_device *dssdev)
{
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1666 1667 1668 1669 1670 1671 1672
	int r;

	mutex_lock(&td->lock);
	r = td->mirror;
	mutex_unlock(&td->lock);

	return r;
1673 1674 1675 1676
}

static int taal_run_test(struct omap_dss_device *dssdev, int test_num)
{
1677
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);
1678 1679 1680
	u8 id1, id2, id3;
	int r;

1681
	mutex_lock(&td->lock);
1682 1683 1684 1685 1686 1687

	if (!td->enabled) {
		r = -ENODEV;
		goto err1;
	}

1688
	dsi_bus_lock(dssdev);
T
Tomi Valkeinen 已提交
1689

1690 1691 1692 1693
	r = taal_wake_up(dssdev);
	if (r)
		goto err2;

1694
	r = taal_dcs_read_1(td, DCS_GET_ID1, &id1);
1695
	if (r)
1696
		goto err2;
1697
	r = taal_dcs_read_1(td, DCS_GET_ID2, &id2);
1698
	if (r)
1699
		goto err2;
1700
	r = taal_dcs_read_1(td, DCS_GET_ID3, &id3);
1701
	if (r)
1702
		goto err2;
1703

1704
	dsi_bus_unlock(dssdev);
1705
	mutex_unlock(&td->lock);
1706
	return 0;
1707
err2:
1708
	dsi_bus_unlock(dssdev);
1709
err1:
1710
	mutex_unlock(&td->lock);
T
Tomi Valkeinen 已提交
1711
	return r;
1712 1713 1714 1715 1716 1717 1718 1719 1720 1721
}

static int taal_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;
T
Tomi Valkeinen 已提交
1722 1723
	struct taal_data *td = dev_get_drvdata(&dssdev->dev);

1724 1725 1726
	if (size < w * h * 3)
		return -ENOMEM;

1727 1728 1729 1730 1731 1732 1733
	mutex_lock(&td->lock);

	if (!td->enabled) {
		r = -ENODEV;
		goto err1;
	}

1734 1735 1736 1737
	size = min(w * h * 3,
			dssdev->panel.timings.x_res *
			dssdev->panel.timings.y_res * 3);

1738
	dsi_bus_lock(dssdev);
T
Tomi Valkeinen 已提交
1739

1740 1741 1742 1743
	r = taal_wake_up(dssdev);
	if (r)
		goto err2;

1744 1745 1746 1747 1748 1749 1750 1751
	/* 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;

1752
	taal_set_update_window(td, x, y, w, h);
1753

1754
	r = dsi_vc_set_max_rx_packet_size(dssdev, td->channel, plen);
1755
	if (r)
1756
		goto err2;
1757 1758 1759 1760 1761

	while (buf_used < size) {
		u8 dcs_cmd = first ? 0x2e : 0x3e;
		first = 0;

1762
		r = dsi_vc_dcs_read(dssdev, td->channel, dcs_cmd,
1763 1764 1765 1766
				buf + buf_used, size - buf_used);

		if (r < 0) {
			dev_err(&dssdev->dev, "read error\n");
1767
			goto err3;
1768 1769 1770 1771 1772 1773 1774 1775 1776 1777 1778 1779 1780
		}

		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;
1781
			goto err3;
1782 1783 1784 1785 1786
		}
	}

	r = buf_used;

1787
err3:
1788
	dsi_vc_set_max_rx_packet_size(dssdev, td->channel, 1);
1789
err2:
1790
	dsi_bus_unlock(dssdev);
1791 1792
err1:
	mutex_unlock(&td->lock);
1793 1794 1795
	return r;
}

1796 1797 1798 1799 1800 1801 1802 1803 1804 1805 1806 1807 1808
static void taal_ulps_work(struct work_struct *work)
{
	struct taal_data *td = container_of(work, struct taal_data,
			ulps_work.work);
	struct omap_dss_device *dssdev = td->dssdev;

	mutex_lock(&td->lock);

	if (dssdev->state != OMAP_DSS_DISPLAY_ACTIVE || !td->enabled) {
		mutex_unlock(&td->lock);
		return;
	}

1809
	dsi_bus_lock(dssdev);
1810 1811 1812

	taal_enter_ulps(dssdev);

1813
	dsi_bus_unlock(dssdev);
1814 1815 1816
	mutex_unlock(&td->lock);
}

1817 1818 1819 1820 1821
static void taal_esd_work(struct work_struct *work)
{
	struct taal_data *td = container_of(work, struct taal_data,
			esd_work.work);
	struct omap_dss_device *dssdev = td->dssdev;
1822
	struct nokia_dsi_panel_data *panel_data = get_panel_data(dssdev);
1823 1824 1825
	u8 state1, state2;
	int r;

1826 1827 1828 1829
	mutex_lock(&td->lock);

	if (!td->enabled) {
		mutex_unlock(&td->lock);
1830
		return;
1831
	}
1832

1833
	dsi_bus_lock(dssdev);
1834

1835 1836 1837 1838 1839 1840
	r = taal_wake_up(dssdev);
	if (r) {
		dev_err(&dssdev->dev, "failed to exit ULPS\n");
		goto err;
	}

1841
	r = taal_dcs_read_1(td, MIPI_DCS_GET_DIAGNOSTIC_RESULT, &state1);
1842 1843 1844 1845 1846 1847 1848 1849 1850 1851 1852 1853
	if (r) {
		dev_err(&dssdev->dev, "failed to read Taal status\n");
		goto err;
	}

	/* Run self diagnostics */
	r = taal_sleep_out(td);
	if (r) {
		dev_err(&dssdev->dev, "failed to run Taal self-diagnostics\n");
		goto err;
	}

1854
	r = taal_dcs_read_1(td, MIPI_DCS_GET_DIAGNOSTIC_RESULT, &state2);
1855 1856 1857 1858 1859 1860 1861 1862 1863 1864 1865 1866 1867 1868
	if (r) {
		dev_err(&dssdev->dev, "failed to read Taal 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 */
1869
	if (td->te_enabled && panel_data->use_ext_te) {
1870
		r = taal_dcs_write_1(td, MIPI_DCS_SET_TEAR_ON, 0);
T
Tomi Valkeinen 已提交
1871 1872 1873
		if (r)
			goto err;
	}
1874

1875
	dsi_bus_unlock(dssdev);
1876

1877
	taal_queue_esd_work(dssdev);
1878

1879
	mutex_unlock(&td->lock);
1880 1881 1882 1883
	return;
err:
	dev_err(&dssdev->dev, "performing LCD reset\n");

1884
	taal_panel_reset(dssdev);
1885

1886
	dsi_bus_unlock(dssdev);
1887

1888
	taal_queue_esd_work(dssdev);
1889 1890

	mutex_unlock(&td->lock);
1891 1892 1893 1894
}

static struct omap_dss_driver taal_driver = {
	.probe		= taal_probe,
1895
	.remove		= __exit_p(taal_remove),
1896 1897 1898 1899 1900 1901

	.enable		= taal_enable,
	.disable	= taal_disable,
	.suspend	= taal_suspend,
	.resume		= taal_resume,

1902 1903 1904
	.update		= taal_update,
	.sync		= taal_sync,

1905
	.get_resolution	= taal_get_resolution,
1906 1907
	.get_recommended_bpp = omapdss_default_get_recommended_bpp,

1908
	.enable_te	= taal_enable_te,
1909 1910
	.get_te		= taal_get_te,

1911 1912 1913 1914 1915 1916 1917
	.set_rotate	= taal_rotate,
	.get_rotate	= taal_get_rotate,
	.set_mirror	= taal_mirror,
	.get_mirror	= taal_get_mirror,
	.run_test	= taal_run_test,
	.memory_read	= taal_memory_read,

1918 1919
	.get_timings	= taal_get_timings,

1920 1921 1922 1923 1924 1925 1926 1927 1928 1929 1930 1931 1932 1933 1934 1935 1936 1937 1938 1939 1940 1941 1942 1943
	.driver         = {
		.name   = "taal",
		.owner  = THIS_MODULE,
	},
};

static int __init taal_init(void)
{
	omap_dss_register_driver(&taal_driver);

	return 0;
}

static void __exit taal_exit(void)
{
	omap_dss_unregister_driver(&taal_driver);
}

module_init(taal_init);
module_exit(taal_exit);

MODULE_AUTHOR("Tomi Valkeinen <tomi.valkeinen@nokia.com>");
MODULE_DESCRIPTION("Taal Driver");
MODULE_LICENSE("GPL");