Root/target/linux/coldfire/patches/018-Add-SSD1289-TFT-LCD-framebuffer-driver-on-TWR-MCF544.patch

1From c1de95e3a608c48b5576e32181480930d9106ac4 Mon Sep 17 00:00:00 2001
2From: Alison Wang <b18965@freescale.com>
3Date: Thu, 4 Aug 2011 09:59:44 +0800
4Subject: [PATCH 18/52] Add SSD1289 TFT LCD framebuffer driver on TWR-MCF5441X
5
6Add SSD1289 TFT LCD framebuffer driver on TWR-MCF5441X. Flexbus and spi
7interfaces are both supported.
8
9Signed-off-by: Alison Wang <b18965@freescale.com>
10---
11 arch/m68k/coldfire/m5441x/config.c | 1 +
12 arch/m68k/coldfire/m5441x/devices.c | 1 +
13 arch/m68k/include/asm/fsl-ssd1289-fb.h | 93 ++++
14 drivers/video/Kconfig | 24 +
15 drivers/video/Makefile | 1 +
16 drivers/video/fsl-ssd1289-fb.c | 791 ++++++++++++++++++++++++++++++++
17 6 files changed, 911 insertions(+), 0 deletions(-)
18 create mode 100644 arch/m68k/include/asm/fsl-ssd1289-fb.h
19 create mode 100644 drivers/video/fsl-ssd1289-fb.c
20
21--- a/arch/m68k/coldfire/m5441x/config.c
22+++ b/arch/m68k/coldfire/m5441x/config.c
23@@ -45,6 +45,7 @@
24 #include <asm/mcf5441x_fbcs.h>
25 #include <asm/mcf5441x_dtim.h>
26 #include <asm/mcf5441x_xbs.h>
27+#include <asm/fsl-ssd1289-fb.h>
28 
29 extern int get_irq_list(struct seq_file *p, void *v);
30 extern char _text, _end;
31--- a/arch/m68k/coldfire/m5441x/devices.c
32+++ b/arch/m68k/coldfire/m5441x/devices.c
33@@ -33,6 +33,7 @@
34 #include <asm/mcfqspi.h>
35 #include <asm/mcfdspi.h>
36 #include <asm/cf_io.h>
37+#include <asm/fsl-ssd1289-fb.h>
38 
39 /*
40  * I2C: only support i2c0 module on m5441x platform
41--- /dev/null
42+++ b/arch/m68k/include/asm/fsl-ssd1289-fb.h
43@@ -0,0 +1,93 @@
44+/*
45+ * Copyright (C) 2010-2011 Freescale Semiconductor, Inc. All Rights Reserved.
46+ *
47+ * Freescale MCF54418 SSD1289 TFT LCD framebuffer driver
48+ *
49+ * This program is free software; you can redistribute it and/or modify it
50+ * under the terms of the GNU General Public License as published by the
51+ * Free Software Foundation; either version 2 of the License, or (at your
52+ * option) any later version.
53+ */
54+
55+#ifndef __FSL_SSD1289_FB_H__
56+#define __FSL_SSD1289_FB_H__
57+
58+#define SSD1289_REG_OSCILLATION 0x00
59+#define SSD1289_REG_DRIVER_OUT_CTRL 0x01
60+#define SSD1289_REG_LCD_DRIVE_AC 0x02
61+#define SSD1289_REG_POWER_CTRL_1 0x03
62+#define SSD1289_REG_COMPARE_1 0x05
63+#define SSD1289_REG_COMPARE_2 0x06
64+#define SSD1289_REG_DISPLAY_CTRL 0x07
65+#define SSD1289_REG_FRAME_CYCLE 0x0b
66+#define SSD1289_REG_POWER_CTRL_2 0x0c
67+#define SSD1289_REG_POWER_CTRL_3 0x0d
68+#define SSD1289_REG_POWER_CTRL_4 0x0e
69+#define SSD1289_REG_GATE_SCAN_START 0x0f
70+#define SSD1289_REG_SLEEP_MODE 0x10
71+#define SSD1289_REG_ENTRY_MODE 0x11
72+#define SSD1289_REG_OPT_SPEED_3 0x12
73+#define SSD1289_REG_H_PORCH 0x16
74+#define SSD1289_REG_V_PORCH 0x17
75+#define SSD1289_REG_POWER_CTRL_5 0x1e
76+#define SSD1289_REG_GDDRAM_DATA 0x22
77+#define SSD1289_REG_WR_DATA_MASK_1 0x23
78+#define SSD1289_REG_WR_DATA_MASK_2 0x24
79+#define SSD1289_REG_FRAME_FREQUENCY 0x25
80+#define SSD1289_REG_OPT_SPEED_1 0x28
81+#define SSD1289_REG_OPT_SPEED_2 0x2f
82+#define SSD1289_REG_GAMMA_CTRL_1 0x30
83+#define SSD1289_REG_GAMMA_CTRL_2 0x31
84+#define SSD1289_REG_GAMMA_CTRL_3 0x32
85+#define SSD1289_REG_GAMMA_CTRL_4 0x33
86+#define SSD1289_REG_GAMMA_CTRL_5 0x34
87+#define SSD1289_REG_GAMMA_CTRL_6 0x35
88+#define SSD1289_REG_GAMMA_CTRL_7 0x36
89+#define SSD1289_REG_GAMMA_CTRL_8 0x37
90+#define SSD1289_REG_GAMMA_CTRL_9 0x3a
91+#define SSD1289_REG_GAMMA_CTRL_10 0x3b
92+#define SSD1289_REG_V_SCROLL_CTRL_1 0x41
93+#define SSD1289_REG_V_SCROLL_CTRL_2 0x42
94+#define SSD1289_REG_H_RAM_ADR_POS 0x44
95+#define SSD1289_REG_V_RAM_ADR_START 0x45
96+#define SSD1289_REG_V_RAM_ADR_END 0x46
97+#define SSD1289_REG_FIRST_WIN_START 0x48
98+#define SSD1289_REG_FIRST_WIN_END 0x49
99+#define SSD1289_REG_SECND_WIN_START 0x4a
100+#define SSD1289_REG_SECND_WIN_END 0x4b
101+#define SSD1289_REG_GDDRAM_X_ADDR 0x4e
102+#define SSD1289_REG_GDDRAM_Y_ADDR 0x4f
103+
104+struct ssd1289 {
105+ void __iomem *cmd;
106+ void __iomem *data;
107+} __packed;
108+
109+struct fsl_ssd1289_fb_info {
110+ struct device *dev;
111+ struct ssd1289 ssd1289_reg;
112+ int openflag;
113+ struct spi_device *spidev;
114+
115+ struct task_struct *task;
116+ unsigned long pseudo_palette[16];
117+};
118+
119+/* LCD description */
120+struct fsl_ssd1289_fb_display {
121+ /* Screen size */
122+ unsigned short width;
123+ unsigned short height;
124+
125+ /* Screen info */
126+ unsigned short xres;
127+ unsigned short yres;
128+ unsigned short bpp;
129+};
130+
131+#define FLEXBUS_LCD_CMD_ADDRESS 0xc0000000
132+#define FLEXBUS_LCD_DATA_ADDRESS 0xc0010000
133+
134+#define SPI_LCD_BLOCK_SIZE 4096
135+#define SPI_LCD_BLOCK_HALF_SIZE 2048
136+#endif
137--- a/drivers/video/Kconfig
138+++ b/drivers/video/Kconfig
139@@ -1980,6 +1980,30 @@ config FB_FSL_DIU
140     ---help---
141       Framebuffer driver for the Freescale SoC DIU
142 
143+config FB_FSL_SSD1289
144+ tristate "SSD1289 TFT LCD (Freescale MCF54418)"
145+ depends on FB && M5441X
146+ select FB_CFB_FILLRECT
147+ select FB_CFB_COPYAREA
148+ select FB_CFB_IMAGEBLIT
149+ select FB_SYS_FOPS
150+ ---help---
151+ This is the framebuffer device driver for a Solomon Systech 240RGBx320
152+ TFT LCD SSD1289.
153+
154+choice
155+ prompt "SSD1289 LCD Controller Interface mode"
156+ depends on FB_FSL_SSD1289
157+
158+config SSD1289_FLEXBUS_MODE
159+ bool "SSD1289 LCD Controller Flexbus Interface mode"
160+
161+config SSD1289_SPI_MODE
162+ bool "SSD1289 LCD Controller SPI Interface mode"
163+ depends on SPI_DSPI && DSPI0
164+
165+endchoice
166+
167 config FB_W100
168     tristate "W100 frame buffer support"
169     depends on FB && ARCH_PXA
170--- a/drivers/video/Makefile
171+++ b/drivers/video/Makefile
172@@ -120,6 +120,7 @@ obj-$(CONFIG_FB_IMX) += imx
173 obj-$(CONFIG_FB_S3C) += s3c-fb.o
174 obj-$(CONFIG_FB_S3C2410) += s3c2410fb.o
175 obj-$(CONFIG_FB_FSL_DIU) += fsl-diu-fb.o
176+obj-$(CONFIG_FB_FSL_SSD1289) += fsl-ssd1289-fb.o
177 obj-$(CONFIG_FB_COBALT) += cobalt_lcdfb.o
178 obj-$(CONFIG_FB_PNX4008_DUM) += pnx4008/
179 obj-$(CONFIG_FB_PNX4008_DUM_RGB) += pnx4008/
180--- /dev/null
181+++ b/drivers/video/fsl-ssd1289-fb.c
182@@ -0,0 +1,791 @@
183+/*
184+ * Copyright (C) 2010-2011 Freescale Semiconductor, Inc. All Rights Reserved.
185+ *
186+ * Freescale MCF54418 SSD1289 TFT LCD framebuffer driver
187+ *
188+ * Author: Alison Wang <b18965@freescale.com>
189+ * Jason Jin <Jason.jin@freescale.com>
190+ *
191+ * This program is free software; you can redistribute it and/or modify it
192+ * under the terms of the GNU General Public License as published by the
193+ * Free Software Foundation; either version 2 of the License, or (at your
194+ * option) any later version.
195+ */
196+
197+#include <linux/module.h>
198+#include <linux/kernel.h>
199+#include <linux/errno.h>
200+#include <linux/string.h>
201+#include <linux/mm.h>
202+#include <linux/slab.h>
203+#include <linux/delay.h>
204+#include <linux/fb.h>
205+#include <linux/init.h>
206+#include <linux/interrupt.h>
207+#include <linux/platform_device.h>
208+#include <linux/dma-mapping.h>
209+#include <linux/clk.h>
210+#include <linux/uaccess.h>
211+#include <linux/timer.h>
212+#include <linux/kthread.h>
213+#include <linux/spi/spi.h>
214+
215+#include <asm/mcf5441x_fbcs.h>
216+#include <asm/mcf5441x_dspi.h>
217+#include <asm/fsl-ssd1289-fb.h>
218+#include <asm/mcf5441x_gpio.h>
219+#include <asm/mcf5441x_ccm.h>
220+#include <asm/mcf_edma.h>
221+
222+#ifdef CONFIG_PM
223+#include <linux/pm.h>
224+#endif
225+
226+#if defined(CONFIG_SSD1289_SPI_MODE)
227+unsigned char spi_block_buffer[SPI_LCD_BLOCK_SIZE];
228+
229+static int ssd1289_spi_writeblock(struct fb_info *info,
230+ unsigned char *daddr, int flag)
231+{
232+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
233+ struct spi_device *devtmp;
234+ int i;
235+
236+ for (i = 0; i < SPI_LCD_BLOCK_SIZE; i++) {
237+ if (i % 2 == 0)
238+ spi_block_buffer[i] = 0x01;
239+ else if (flag == 1)
240+ spi_block_buffer[i] = *(daddr + (i >> 1));
241+ else if (flag == 0)
242+ spi_block_buffer[i] = 0;
243+ }
244+
245+ devtmp = fbinfo->spidev;
246+ spi_write(devtmp, (const unsigned char *)spi_block_buffer,
247+ SPI_LCD_BLOCK_SIZE);
248+ return 0;
249+}
250+
251+static int ssd1289_spi_write(struct fb_info *info,
252+ unsigned short value, unsigned int flag)
253+{
254+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
255+ struct spi_device *devtmp;
256+ unsigned short tmpl;
257+ unsigned short tmph;
258+
259+ devtmp = fbinfo->spidev;
260+ if (flag == 1) {
261+ /* D/C = 1 */
262+ tmph = ((value >> 8) & 0xff) + 0x0100;
263+ tmpl = (value & 0xff) + 0x0100;
264+ spi_write(devtmp, (const u8 *)&tmph, sizeof(tmph));
265+ spi_write(devtmp, (const u8 *)&tmpl, sizeof(tmpl));
266+ } else {
267+ /* D/C = 0 */
268+ tmpl = (value & 0xff);
269+ spi_write(devtmp, (const u8 *)&tmpl, sizeof(tmpl));
270+ }
271+ return 0;
272+}
273+#elif defined(CONFIG_SSD1289_FLEXBUS_MODE)
274+static int ssd1289_flexbus_write(struct fb_info *info, unsigned short value,
275+ unsigned int flag)
276+{
277+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
278+ void __iomem *cmd_addr, *data_addr;
279+
280+ cmd_addr = fbinfo->ssd1289_reg.cmd;
281+ data_addr = fbinfo->ssd1289_reg.data;
282+
283+ if (flag == 0)
284+ out_be16(cmd_addr, value);
285+ else
286+ out_be16(data_addr, value);
287+
288+ return 0;
289+}
290+#endif
291+
292+static int ssd1289_write(struct fb_info *info, unsigned short value,
293+ unsigned int flag)
294+{
295+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
296+ ssd1289_flexbus_write(info, value, flag);
297+#elif defined(CONFIG_SSD1289_SPI_MODE)
298+ ssd1289_spi_write(info, value, flag);
299+#endif
300+ return 0;
301+}
302+
303+static void fsl_ssd1289_enable_lcd(struct fb_info *info)
304+{
305+ int i;
306+
307+#if defined(CONFIG_SSD1289_SPI_MODE)
308+ int count;
309+#elif defined(CONFIG_SSD1289_FLEXBUS_MODE)
310+ /* GPIO configuration */
311+ MCF_GPIO_PAR_BE = MCF_GPIO_PAR_BE_BE3_FB_A1 | MCF_GPIO_PAR_BE_BE2_FB_A0
312+ | MCF_GPIO_PAR_BE_BE1_BE1 | MCF_GPIO_PAR_BE_BE0_BE0;
313+ MCF_GPIO_PAR_CS |= MCF_GPIO_PAR_CS_CS0_CS0;
314+#endif
315+
316+ ssd1289_write(info, SSD1289_REG_DISPLAY_CTRL, 0);
317+ ssd1289_write(info, 0x0200, 1);
318+
319+ ssd1289_write(info, SSD1289_REG_OSCILLATION, 0);
320+ ssd1289_write(info, 0x0000, 1);
321+
322+ mdelay(100);
323+
324+ /* turn on the oscillator */
325+ ssd1289_write(info, SSD1289_REG_OSCILLATION, 0);
326+ ssd1289_write(info, 0x0001, 1);
327+
328+ mdelay(100);
329+ /* power control 1 */
330+ ssd1289_write(info, SSD1289_REG_POWER_CTRL_1, 0);
331+ ssd1289_write(info, 0xaeac, 1);
332+
333+ /* power control 2 */
334+ ssd1289_write(info, SSD1289_REG_POWER_CTRL_2, 0);
335+ ssd1289_write(info, 0x0007, 1);
336+
337+ /* power control 3 */
338+ ssd1289_write(info, SSD1289_REG_POWER_CTRL_3, 0);
339+ ssd1289_write(info, 0x000f, 1);
340+
341+ /* power control 4 */
342+ ssd1289_write(info, SSD1289_REG_POWER_CTRL_4, 0);
343+ ssd1289_write(info, 0x2900, 1);
344+
345+ /* power control 5 */
346+ ssd1289_write(info, SSD1289_REG_POWER_CTRL_5, 0);
347+ ssd1289_write(info, 0x00b3, 1);
348+
349+ mdelay(15);
350+ /* driver output control */
351+ ssd1289_write(info, SSD1289_REG_DRIVER_OUT_CTRL, 0);
352+ ssd1289_write(info, 0x2b3f, 1);
353+
354+ /* lcd-driving-waveform control */
355+ ssd1289_write(info, SSD1289_REG_LCD_DRIVE_AC, 0);
356+ ssd1289_write(info, 0x0600, 1);
357+
358+ /* sleep mode */
359+ ssd1289_write(info, SSD1289_REG_SLEEP_MODE, 0);
360+ ssd1289_write(info, 0x0000, 1);
361+
362+ /* entry mode */
363+ ssd1289_write(info, SSD1289_REG_ENTRY_MODE, 0);
364+ ssd1289_write(info, 0x60a8, 1);
365+
366+ mdelay(15);
367+ /* compare register */
368+ ssd1289_write(info, SSD1289_REG_COMPARE_1, 0);
369+ ssd1289_write(info, 0x0000, 1);
370+
371+ ssd1289_write(info, SSD1289_REG_COMPARE_2, 0);
372+ ssd1289_write(info, 0x0000, 1);
373+
374+ /* horizontal porch */
375+ ssd1289_write(info, SSD1289_REG_H_PORCH, 0);
376+ ssd1289_write(info, 0xef1c, 1);
377+
378+ /* vertical porch */
379+ ssd1289_write(info, SSD1289_REG_V_PORCH, 0);
380+ ssd1289_write(info, 0x0003, 1);
381+
382+ /* display control */
383+ ssd1289_write(info, SSD1289_REG_DISPLAY_CTRL, 0);
384+ ssd1289_write(info, 0x0233, 1);
385+
386+ /* frame cycle control */
387+ ssd1289_write(info, SSD1289_REG_FRAME_CYCLE, 0);
388+ ssd1289_write(info, 0x5312, 1);
389+
390+ /* gate scan position */
391+ ssd1289_write(info, SSD1289_REG_GATE_SCAN_START, 0);
392+ ssd1289_write(info, 0x0000, 1);
393+
394+ mdelay(20);
395+ /* vertical scroll control */
396+ ssd1289_write(info, SSD1289_REG_V_SCROLL_CTRL_1, 0);
397+ ssd1289_write(info, 0x0000, 1);
398+
399+ ssd1289_write(info, SSD1289_REG_V_SCROLL_CTRL_2, 0);
400+ ssd1289_write(info, 0x0000, 1);
401+
402+ /* 1st screen driving position */
403+ ssd1289_write(info, SSD1289_REG_FIRST_WIN_START, 0);
404+ ssd1289_write(info, 0x0000, 1);
405+
406+ ssd1289_write(info, SSD1289_REG_FIRST_WIN_END, 0);
407+ ssd1289_write(info, 0x013F, 1);
408+
409+ /* 2nd screen driving position */
410+ ssd1289_write(info, SSD1289_REG_SECND_WIN_START, 0);
411+ ssd1289_write(info, 0x0000, 1);
412+
413+ ssd1289_write(info, SSD1289_REG_SECND_WIN_END, 0);
414+ ssd1289_write(info, 0x0000, 1);
415+
416+ mdelay(20);
417+ /* gamma control */
418+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_1, 0);
419+ ssd1289_write(info, 0x0707, 1);
420+
421+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_2, 0);
422+ ssd1289_write(info, 0x0704, 1);
423+
424+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_3, 0);
425+ ssd1289_write(info, 0x0204, 1);
426+
427+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_4, 0);
428+ ssd1289_write(info, 0x0201, 1);
429+
430+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_5, 0);
431+ ssd1289_write(info, 0x0203, 1);
432+
433+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_6, 0);
434+ ssd1289_write(info, 0x0204, 1);
435+
436+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_7, 0);
437+ ssd1289_write(info, 0x0204, 1);
438+
439+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_8, 0);
440+ ssd1289_write(info, 0x0502, 1);
441+
442+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_9, 0);
443+ ssd1289_write(info, 0x0302, 1);
444+
445+ ssd1289_write(info, SSD1289_REG_GAMMA_CTRL_10, 0);
446+ ssd1289_write(info, 0x0500, 1);
447+
448+ /* ram write data mask */
449+ ssd1289_write(info, SSD1289_REG_WR_DATA_MASK_1, 0);
450+ ssd1289_write(info, 0x0000, 1);
451+
452+ ssd1289_write(info, SSD1289_REG_WR_DATA_MASK_2, 0);
453+ ssd1289_write(info, 0x0000, 1);
454+
455+ /* frame frequency control */
456+ ssd1289_write(info, SSD1289_REG_FRAME_FREQUENCY, 0);
457+ ssd1289_write(info, 0xe000, 1);
458+
459+ /* optimize data access speed */
460+ ssd1289_write(info, SSD1289_REG_OPT_SPEED_1, 0);
461+ ssd1289_write(info, 0x0006, 1);
462+
463+ ssd1289_write(info, SSD1289_REG_OPT_SPEED_2, 0);
464+ ssd1289_write(info, 0x12ae, 1);
465+
466+ ssd1289_write(info, SSD1289_REG_OPT_SPEED_3, 0);
467+ ssd1289_write(info, 0x6ceb, 1);
468+
469+ /* horizontal ram address position */
470+ ssd1289_write(info, SSD1289_REG_H_RAM_ADR_POS, 0);
471+ ssd1289_write(info, 0xef00, 1);
472+
473+ /* vertical ram address position */
474+ ssd1289_write(info, SSD1289_REG_V_RAM_ADR_START, 0);
475+ ssd1289_write(info, 0x0000, 1);
476+
477+ ssd1289_write(info, SSD1289_REG_V_RAM_ADR_END, 0);
478+ ssd1289_write(info, 0x013f, 1);
479+
480+ mdelay(20);
481+
482+ /* set start address counter */
483+ ssd1289_write(info, SSD1289_REG_GDDRAM_X_ADDR, 0);
484+ ssd1289_write(info, 0x00ef, 1);
485+
486+ ssd1289_write(info, SSD1289_REG_GDDRAM_Y_ADDR, 0);
487+ ssd1289_write(info, 0x0000, 1);
488+
489+ ssd1289_write(info, SSD1289_REG_GDDRAM_DATA, 0);
490+
491+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
492+ for (i = 0; i < info->screen_size; i += 2)
493+ ssd1289_write(info, 0, 1);
494+#elif defined(CONFIG_SSD1289_SPI_MODE)
495+ count = info->screen_size / SPI_LCD_BLOCK_HALF_SIZE;
496+ for (i = 0; i < count; i++)
497+ ssd1289_spi_writeblock(info, NULL, 0);
498+#endif
499+}
500+
501+static void fsl_ssd1289_disable_lcd(struct fb_info *info)
502+{
503+ ssd1289_write(info, SSD1289_REG_DISPLAY_CTRL, 0);
504+ ssd1289_write(info, 0x0200, 1);
505+
506+ ssd1289_write(info, SSD1289_REG_OSCILLATION, 0);
507+ ssd1289_write(info, 0x0000, 1);
508+}
509+
510+static int ssd1289fbd(void *arg)
511+{
512+ struct fb_info *info = arg;
513+ int i;
514+ unsigned short *buf_p;
515+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
516+#if defined(CONFIG_SSD1289_SPI_MODE)
517+ unsigned char *bufspi_p;
518+ int count;
519+#endif
520+
521+ while (!kthread_should_stop()) {
522+ set_current_state(TASK_INTERRUPTIBLE);
523+
524+ if (fbinfo->openflag == 1) {
525+ buf_p = (unsigned short *)(info->screen_base);
526+
527+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
528+ for (i = 0; i < info->screen_size; i += 2) {
529+ ssd1289_write(info, *buf_p, 1);
530+ buf_p++;
531+ }
532+#elif defined(CONFIG_SSD1289_SPI_MODE)
533+ bufspi_p = (unsigned char *)buf_p;
534+ count = info->screen_size / SPI_LCD_BLOCK_HALF_SIZE;
535+ for (i = 0; i < count; i++)
536+ ssd1289_spi_writeblock(info, (bufspi_p +
537+ SPI_LCD_BLOCK_HALF_SIZE * i), 1);
538+#endif
539+ }
540+ schedule_timeout(HZ/25);
541+ }
542+
543+ return 0;
544+}
545+
546+static int fsl_ssd1289_check_var(struct fb_var_screeninfo *var,
547+ struct fb_info *info)
548+{
549+ if (var->xres_virtual < var->xres)
550+ var->xres_virtual = var->xres;
551+ if (var->yres_virtual < var->yres)
552+ var->yres_virtual = var->yres;
553+
554+ if (var->xoffset < 0)
555+ var->xoffset = 0;
556+
557+ if (var->yoffset < 0)
558+ var->yoffset = 0;
559+
560+ if (var->xoffset + info->var.xres > info->var.xres_virtual)
561+ var->xoffset = info->var.xres_virtual - info->var.xres;
562+
563+ if (var->yoffset + info->var.yres > info->var.yres_virtual)
564+ var->yoffset = info->var.yres_virtual - info->var.yres;
565+
566+ switch (var->bits_per_pixel) {
567+ case 8:
568+ /* 8 bpp, 332 format */
569+ var->red.length = 3;
570+ var->red.offset = 5;
571+ var->red.msb_right = 0;
572+
573+ var->green.length = 3;
574+ var->green.offset = 2;
575+ var->green.msb_right = 0;
576+
577+ var->blue.length = 2;
578+ var->blue.offset = 0;
579+ var->blue.msb_right = 0;
580+
581+ var->transp.length = 0;
582+ var->transp.offset = 0;
583+ var->transp.msb_right = 0;
584+ break;
585+ case 16:
586+ /* 16 bpp, 565 format */
587+ var->red.length = 5;
588+ var->red.offset = 11;
589+ var->red.msb_right = 0;
590+
591+ var->green.length = 6;
592+ var->green.offset = 5;
593+ var->green.msb_right = 0;
594+
595+ var->blue.length = 5;
596+ var->blue.offset = 0;
597+ var->blue.msb_right = 0;
598+
599+ var->transp.length = 0;
600+ var->transp.offset = 0;
601+ var->transp.msb_right = 0;
602+ break;
603+ default:
604+ printk(KERN_ERR "Depth not supported: %u BPP\n",
605+ var->bits_per_pixel);
606+ return -EINVAL;
607+ }
608+ return 0;
609+}
610+
611+static int fsl_ssd1289_set_par(struct fb_info *info)
612+{
613+ struct fb_var_screeninfo *var = &info->var;
614+
615+ switch (var->bits_per_pixel) {
616+ case 16:
617+ info->fix.visual = FB_VISUAL_TRUECOLOR;
618+ break;
619+ case 8:
620+ info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
621+ break;
622+ default:
623+ info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
624+ break;
625+ }
626+
627+ return 0;
628+}
629+
630+static inline __u32 CNVT_TOHW(__u32 val, __u32 width)
631+{
632+ return ((val<<width) + 0x7FFF - val)>>16;
633+}
634+
635+static int fsl_ssd1289_setcolreg(unsigned regno,
636+ unsigned red, unsigned green, unsigned blue,
637+ unsigned transp, struct fb_info *info)
638+{
639+ int ret = 1;
640+
641+ /*
642+ * If greyscale is true, then we convert the RGB value
643+ * to greyscale no matter what visual we are using.
644+ */
645+ if (info->var.grayscale)
646+ red = green = blue = (19595 * red + 38470 * green +
647+ 7471 * blue) >> 16;
648+ switch (info->fix.visual) {
649+ case FB_VISUAL_TRUECOLOR:
650+ if (regno < 16) {
651+ u32 *pal = info->pseudo_palette;
652+ u32 value;
653+
654+ red = CNVT_TOHW(red, info->var.red.length);
655+ green = CNVT_TOHW(green, info->var.green.length);
656+ blue = CNVT_TOHW(blue, info->var.blue.length);
657+ transp = CNVT_TOHW(transp, info->var.transp.length);
658+
659+ value = (red << info->var.red.offset) |
660+ (green << info->var.green.offset) |
661+ (blue << info->var.blue.offset) |
662+ (transp << info->var.transp.offset);
663+
664+ pal[regno] = value;
665+ ret = 0;
666+ }
667+ break;
668+ case FB_VISUAL_STATIC_PSEUDOCOLOR:
669+ case FB_VISUAL_PSEUDOCOLOR:
670+ break;
671+ }
672+ return ret;
673+}
674+
675+static int fsl_ssd1289_blank(int blank_mode, struct fb_info *info)
676+{
677+ if (blank_mode == FB_BLANK_POWERDOWN)
678+ fsl_ssd1289_disable_lcd(info);
679+ else
680+ fsl_ssd1289_enable_lcd(info);
681+
682+ return 0;
683+}
684+
685+static int fsl_ssd1289_open(struct fb_info *info, int user)
686+{
687+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
688+ struct task_struct *task;
689+ int ret;
690+
691+ if (fbinfo->openflag == 0) {
692+ memset(info->screen_base, 0, info->screen_size);
693+ fsl_ssd1289_enable_lcd(info);
694+
695+ task = kthread_run(ssd1289fbd, info, "SSD1289 LCD");
696+ if (IS_ERR(task)) {
697+ ret = PTR_ERR(task);
698+ return ret;
699+ }
700+ fbinfo->task = task;
701+ }
702+
703+ fbinfo->openflag = 1;
704+ return 0;
705+}
706+
707+static int fsl_ssd1289_release(struct fb_info *info, int user)
708+{
709+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
710+
711+ fbinfo->openflag = 0;
712+ if (fbinfo->task) {
713+ struct task_struct *task = fbinfo->task;
714+ fbinfo->task = NULL;
715+ kthread_stop(task);
716+ }
717+
718+ memset(info->screen_base, 0, info->screen_size);
719+ fsl_ssd1289_disable_lcd(info);
720+ return 0;
721+}
722+
723+static struct fb_ops fsl_ssd1289_ops = {
724+ .owner = THIS_MODULE,
725+ .fb_check_var = fsl_ssd1289_check_var,
726+ .fb_set_par = fsl_ssd1289_set_par,
727+ .fb_setcolreg = fsl_ssd1289_setcolreg,
728+ .fb_blank = fsl_ssd1289_blank,
729+ .fb_open = fsl_ssd1289_open,
730+ .fb_release = fsl_ssd1289_release,
731+ .fb_copyarea = cfb_copyarea,
732+ .fb_fillrect = cfb_fillrect,
733+ .fb_imageblit = cfb_imageblit,
734+};
735+
736+static int fsl_ssd1289_map_video_memory(struct fb_info *info)
737+{
738+ unsigned int map_size = info->fix.smem_len;
739+
740+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
741+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
742+
743+ fbinfo->ssd1289_reg.cmd =
744+ ioremap_nocache(FLEXBUS_LCD_CMD_ADDRESS, 2);
745+ fbinfo->ssd1289_reg.data =
746+ ioremap_nocache(FLEXBUS_LCD_DATA_ADDRESS, 2);
747+#endif
748+
749+ info->screen_base = kmalloc(map_size, GFP_KERNEL);
750+ info->fix.smem_start = virt_to_phys(info->screen_base);
751+ info->screen_size = info->fix.smem_len;
752+
753+ if (info->screen_base)
754+ memset(info->screen_base, 0, map_size);
755+
756+ return info->screen_base ? 0 : -ENOMEM;
757+}
758+
759+static inline void fsl_ssd1289_unmap_video_memory(struct fb_info *info)
760+{
761+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
762+ struct fsl_ssd1289_fb_info *fbinfo = info->par;
763+
764+ iounmap(fbinfo->ssd1289_reg.cmd);
765+ iounmap(fbinfo->ssd1289_reg.data);
766+#endif
767+ kfree(info->screen_base);
768+}
769+
770+
771+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
772+static int fsl_ssd1289_probe(struct platform_device *pdev)
773+#elif defined(CONFIG_SSD1289_SPI_MODE)
774+static int fsl_ssd1289_probe(struct spi_device *spi)
775+#endif
776+{
777+ struct fsl_ssd1289_fb_info *fbinfo;
778+ struct fb_info *info;
779+ struct fsl_ssd1289_fb_display *display;
780+ int ret;
781+ unsigned long smem_len;
782+
783+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
784+ info = framebuffer_alloc(sizeof(struct fsl_ssd1289_fb_info),
785+ &pdev->dev);
786+ if (!info)
787+ return -ENOMEM;
788+
789+ platform_set_drvdata(pdev, info);
790+
791+ fbinfo = info->par;
792+ fbinfo->dev = &pdev->dev;
793+ display = pdev->dev.platform_data;
794+#elif defined(CONFIG_SSD1289_SPI_MODE)
795+ info = framebuffer_alloc(sizeof(struct fsl_ssd1289_fb_info),
796+ &spi->dev);
797+ if (!info)
798+ return -ENOMEM;
799+
800+ dev_set_drvdata(&spi->dev, info);
801+
802+ fbinfo = info->par;
803+ fbinfo->dev = &spi->dev;
804+ fbinfo->spidev = spi;
805+ display = spi->dev.platform_data;
806+#endif
807+
808+ fbinfo->openflag = 0;
809+ info->fix.type = FB_TYPE_PACKED_PIXELS;
810+ info->fix.type_aux = 0;
811+ info->fix.xpanstep = 0;
812+ info->fix.ypanstep = 0;
813+ info->fix.ywrapstep = 0;
814+ info->fix.accel = FB_ACCEL_NONE;
815+
816+ info->var.nonstd = 0;
817+ info->var.activate = FB_ACTIVATE_NOW;
818+ info->var.accel_flags = 0;
819+ info->var.vmode = FB_VMODE_NONINTERLACED;
820+
821+ info->fbops = &fsl_ssd1289_ops;
822+ info->flags = FBINFO_HWACCEL_IMAGEBLIT | FBINFO_HWACCEL_FILLRECT
823+ | FBINFO_HWACCEL_COPYAREA;
824+ info->pseudo_palette = &fbinfo->pseudo_palette;
825+
826+ /* find maximum required memory size for display */
827+ smem_len = display->xres;
828+ smem_len *= display->yres;
829+ smem_len *= display->bpp;
830+ smem_len >>= 3;
831+ if (info->fix.smem_len < smem_len)
832+ info->fix.smem_len = smem_len;
833+
834+ /* Intialize video memory */
835+ ret = fsl_ssd1289_map_video_memory(info);
836+ if (ret) {
837+ printk(KERN_ERR "Failed to allocate video RAM: %d\n", ret);
838+ ret = -ENOMEM;
839+ goto dealloc_fb;
840+ }
841+
842+ info->var.xres = display->xres;
843+ info->var.yres = display->yres;
844+ info->var.bits_per_pixel = display->bpp;
845+ info->fix.line_length = (info->var.xres * info->var.bits_per_pixel) / 8;
846+
847+ fsl_ssd1289_check_var(&info->var, info);
848+
849+ ret = register_framebuffer(info);
850+ if (ret < 0) {
851+ printk(KERN_ERR "Failed to register framebuffer device: %d\n",
852+ ret);
853+ goto free_video_memory;
854+ }
855+
856+ printk(KERN_INFO "fb: SSD1289 TFT LCD Framebuffer Driver\n");
857+ return 0;
858+
859+free_video_memory:
860+ fsl_ssd1289_unmap_video_memory(info);
861+dealloc_fb:
862+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
863+ platform_set_drvdata(pdev, NULL);
864+#elif defined(CONFIG_SSD1289_SPI_MODE)
865+ dev_set_drvdata(&spi->dev, NULL);
866+#endif
867+ framebuffer_release(info);
868+ return ret;
869+}
870+
871+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
872+static int fsl_ssd1289_remove(struct platform_device *pdev)
873+{
874+ struct fb_info *info = platform_get_drvdata(pdev);
875+
876+ platform_set_drvdata(pdev, NULL);
877+#elif defined(CONFIG_SSD1289_SPI_MODE)
878+static int fsl_ssd1289_remove(struct spi_device *spi)
879+{
880+ struct fb_info *info = dev_get_drvdata(&spi->dev);
881+
882+ dev_set_drvdata(&spi->dev, NULL);
883+#endif
884+ unregister_framebuffer(info);
885+ fsl_ssd1289_unmap_video_memory(info);
886+ framebuffer_release(info);
887+ return 0;
888+}
889+
890+#ifdef CONFIG_PM
891+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
892+static int fsl_ssd1289_suspend(struct platform_device *dev, pm_message_t state)
893+{
894+ struct fb_info *info = platform_get_drvdata(dev);
895+#elif defined(CONFIG_SSD1289_SPI_MODE)
896+static int fsl_ssd1289_suspend(struct spi_device *spi, pm_message_t state)
897+{
898+ struct fb_info *info = dev_get_drvdata(&spi->dev);
899+#endif
900+ /* enter into sleep mode */
901+ ssd1289_write(info, SSD1289_REG_SLEEP_MODE, 0);
902+ ssd1289_write(info, 0x0001, 1);
903+ return 0;
904+}
905+
906+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
907+static int fsl_ssd1289_resume(struct platform_device *dev)
908+{
909+ struct fb_info *info = platform_get_drvdata(dev);
910+#elif defined(CONFIG_SSD1289_SPI_MODE)
911+static int fsl_ssd1289_resume(struct spi_device *spi)
912+{
913+ struct fb_info *info = dev_get_drvdata(&spi->dev);
914+#endif
915+ /* leave sleep mode */
916+ ssd1289_write(info, SSD1289_REG_SLEEP_MODE, 0);
917+ ssd1289_write(info, 0x0000, 1);
918+ return 0;
919+}
920+#else
921+#define fsl_ssd1289_suspend NULL
922+#define fsl_ssd1289_resume NULL
923+#endif
924+
925+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
926+static struct platform_driver fsl_ssd1289_driver = {
927+ .probe = fsl_ssd1289_probe,
928+ .remove = fsl_ssd1289_remove,
929+ .suspend = fsl_ssd1289_suspend,
930+ .resume = fsl_ssd1289_resume,
931+ .driver = {
932+ .name = "fsl-ssd1289",
933+ .owner = THIS_MODULE,
934+ },
935+};
936+#elif defined(CONFIG_SSD1289_SPI_MODE)
937+static struct spi_driver spi_ssd1289_driver = {
938+ .driver = {
939+ .name = "spi-ssd1289",
940+ .bus = &spi_bus_type,
941+ .owner = THIS_MODULE,
942+ },
943+ .probe = fsl_ssd1289_probe,
944+ .remove = fsl_ssd1289_remove,
945+ .suspend = fsl_ssd1289_suspend,
946+ .resume = fsl_ssd1289_resume,
947+};
948+#endif
949+
950+static int __devinit fsl_ssd1289_init(void)
951+{
952+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
953+ return platform_driver_register(&fsl_ssd1289_driver);
954+#elif defined(CONFIG_SSD1289_SPI_MODE)
955+ return spi_register_driver(&spi_ssd1289_driver);
956+#endif
957+}
958+
959+static void __exit fsl_ssd1289_exit(void)
960+{
961+#if defined(CONFIG_SSD1289_FLEXBUS_MODE)
962+ return platform_driver_unregister(&fsl_ssd1289_driver);
963+#elif defined(CONFIG_SSD1289_SPI_MODE)
964+ return spi_unregister_driver(&spi_ssd1289_driver);
965+#endif
966+}
967+
968+module_init(fsl_ssd1289_init);
969+module_exit(fsl_ssd1289_exit);
970+
971+MODULE_AUTHOR("Alison Wang <b18965@freescale.com>");
972+MODULE_DESCRIPTION("Freescale MCF54418 SSD1289 TFT LCD Framebuffer Driver");
973+MODULE_LICENSE("GPL");
974

Archive Download this file



interactive