1/*
2 * linux/drivers/video/pxa168fb.c -- Marvell PXA168 LCD Controller
3 *
4 *  Copyright (C) 2008 Marvell International Ltd.
5 *  All rights reserved.
6 *
7 *  2009-02-16  adapted from original version for PXA168/910
8 *              Jun Nie <njun@marvell.com>
9 *
10 * This file is subject to the terms and conditions of the GNU General Public
11 * License. See the file COPYING in the main directory of this archive for
12 * more details.
13 */
14
15#include <linux/module.h>
16#include <linux/kernel.h>
17#include <linux/sched.h>
18#include <linux/string.h>
19#include <linux/interrupt.h>
20#include <linux/slab.h>
21#include <linux/fb.h>
22#include <linux/delay.h>
23#include <linux/init.h>
24#include <linux/io.h>
25#include <linux/ioport.h>
26#include <linux/platform_device.h>
27#include <linux/dma-mapping.h>
28#include <linux/clk.h>
29#include <linux/err.h>
30#include <linux/uaccess.h>
31#include <video/pxa168fb.h>
32
33#include "pxa168fb.h"
34
35#define DEFAULT_REFRESH		60	/* Hz */
36
37static int determine_best_pix_fmt(struct fb_var_screeninfo *var)
38{
39	/*
40	 * Pseudocolor mode?
41	 */
42	if (var->bits_per_pixel == 8)
43		return PIX_FMT_PSEUDOCOLOR;
44
45	/*
46	 * Check for 565/1555.
47	 */
48	if (var->bits_per_pixel == 16 && var->red.length <= 5 &&
49	    var->green.length <= 6 && var->blue.length <= 5) {
50		if (var->transp.length == 0) {
51			if (var->red.offset >= var->blue.offset)
52				return PIX_FMT_RGB565;
53			else
54				return PIX_FMT_BGR565;
55		}
56
57		if (var->transp.length == 1 && var->green.length <= 5) {
58			if (var->red.offset >= var->blue.offset)
59				return PIX_FMT_RGB1555;
60			else
61				return PIX_FMT_BGR1555;
62		}
63	}
64
65	/*
66	 * Check for 888/A888.
67	 */
68	if (var->bits_per_pixel <= 32 && var->red.length <= 8 &&
69	    var->green.length <= 8 && var->blue.length <= 8) {
70		if (var->bits_per_pixel == 24 && var->transp.length == 0) {
71			if (var->red.offset >= var->blue.offset)
72				return PIX_FMT_RGB888PACK;
73			else
74				return PIX_FMT_BGR888PACK;
75		}
76
77		if (var->bits_per_pixel == 32 && var->transp.length == 8) {
78			if (var->red.offset >= var->blue.offset)
79				return PIX_FMT_RGBA888;
80			else
81				return PIX_FMT_BGRA888;
82		} else {
83			if (var->red.offset >= var->blue.offset)
84				return PIX_FMT_RGB888UNPACK;
85			else
86				return PIX_FMT_BGR888UNPACK;
87		}
88	}
89
90	return -EINVAL;
91}
92
93static void set_pix_fmt(struct fb_var_screeninfo *var, int pix_fmt)
94{
95	switch (pix_fmt) {
96	case PIX_FMT_RGB565:
97		var->bits_per_pixel = 16;
98		var->red.offset = 11;    var->red.length = 5;
99		var->green.offset = 5;   var->green.length = 6;
100		var->blue.offset = 0;    var->blue.length = 5;
101		var->transp.offset = 0;  var->transp.length = 0;
102		break;
103	case PIX_FMT_BGR565:
104		var->bits_per_pixel = 16;
105		var->red.offset = 0;     var->red.length = 5;
106		var->green.offset = 5;   var->green.length = 6;
107		var->blue.offset = 11;   var->blue.length = 5;
108		var->transp.offset = 0;  var->transp.length = 0;
109		break;
110	case PIX_FMT_RGB1555:
111		var->bits_per_pixel = 16;
112		var->red.offset = 10;    var->red.length = 5;
113		var->green.offset = 5;   var->green.length = 5;
114		var->blue.offset = 0;    var->blue.length = 5;
115		var->transp.offset = 15; var->transp.length = 1;
116		break;
117	case PIX_FMT_BGR1555:
118		var->bits_per_pixel = 16;
119		var->red.offset = 0;     var->red.length = 5;
120		var->green.offset = 5;   var->green.length = 5;
121		var->blue.offset = 10;   var->blue.length = 5;
122		var->transp.offset = 15; var->transp.length = 1;
123		break;
124	case PIX_FMT_RGB888PACK:
125		var->bits_per_pixel = 24;
126		var->red.offset = 16;    var->red.length = 8;
127		var->green.offset = 8;   var->green.length = 8;
128		var->blue.offset = 0;    var->blue.length = 8;
129		var->transp.offset = 0;  var->transp.length = 0;
130		break;
131	case PIX_FMT_BGR888PACK:
132		var->bits_per_pixel = 24;
133		var->red.offset = 0;     var->red.length = 8;
134		var->green.offset = 8;   var->green.length = 8;
135		var->blue.offset = 16;   var->blue.length = 8;
136		var->transp.offset = 0;  var->transp.length = 0;
137		break;
138	case PIX_FMT_RGBA888:
139		var->bits_per_pixel = 32;
140		var->red.offset = 16;    var->red.length = 8;
141		var->green.offset = 8;   var->green.length = 8;
142		var->blue.offset = 0;    var->blue.length = 8;
143		var->transp.offset = 24; var->transp.length = 8;
144		break;
145	case PIX_FMT_BGRA888:
146		var->bits_per_pixel = 32;
147		var->red.offset = 0;     var->red.length = 8;
148		var->green.offset = 8;   var->green.length = 8;
149		var->blue.offset = 16;   var->blue.length = 8;
150		var->transp.offset = 24; var->transp.length = 8;
151		break;
152	case PIX_FMT_PSEUDOCOLOR:
153		var->bits_per_pixel = 8;
154		var->red.offset = 0;     var->red.length = 8;
155		var->green.offset = 0;   var->green.length = 8;
156		var->blue.offset = 0;    var->blue.length = 8;
157		var->transp.offset = 0;  var->transp.length = 0;
158		break;
159	}
160}
161
162static void set_mode(struct pxa168fb_info *fbi, struct fb_var_screeninfo *var,
163		     struct fb_videomode *mode, int pix_fmt, int ystretch)
164{
165	struct fb_info *info = fbi->info;
166
167	set_pix_fmt(var, pix_fmt);
168
169	var->xres = mode->xres;
170	var->yres = mode->yres;
171	var->xres_virtual = max(var->xres, var->xres_virtual);
172	if (ystretch)
173		var->yres_virtual = info->fix.smem_len /
174			(var->xres_virtual * (var->bits_per_pixel >> 3));
175	else
176		var->yres_virtual = max(var->yres, var->yres_virtual);
177	var->grayscale = 0;
178	var->accel_flags = FB_ACCEL_NONE;
179	var->pixclock = mode->pixclock;
180	var->left_margin = mode->left_margin;
181	var->right_margin = mode->right_margin;
182	var->upper_margin = mode->upper_margin;
183	var->lower_margin = mode->lower_margin;
184	var->hsync_len = mode->hsync_len;
185	var->vsync_len = mode->vsync_len;
186	var->sync = mode->sync;
187	var->vmode = FB_VMODE_NONINTERLACED;
188	var->rotate = FB_ROTATE_UR;
189}
190
191static int pxa168fb_check_var(struct fb_var_screeninfo *var,
192			      struct fb_info *info)
193{
194	struct pxa168fb_info *fbi = info->par;
195	int pix_fmt;
196
197	/*
198	 * Determine which pixel format we're going to use.
199	 */
200	pix_fmt = determine_best_pix_fmt(var);
201	if (pix_fmt < 0)
202		return pix_fmt;
203	set_pix_fmt(var, pix_fmt);
204	fbi->pix_fmt = pix_fmt;
205
206	/*
207	 * Basic geometry sanity checks.
208	 */
209	if (var->xoffset + var->xres > var->xres_virtual)
210		return -EINVAL;
211	if (var->yoffset + var->yres > var->yres_virtual)
212		return -EINVAL;
213	if (var->xres + var->right_margin +
214	    var->hsync_len + var->left_margin > 2048)
215		return -EINVAL;
216	if (var->yres + var->lower_margin +
217	    var->vsync_len + var->upper_margin > 2048)
218		return -EINVAL;
219
220	/*
221	 * Check size of framebuffer.
222	 */
223	if (var->xres_virtual * var->yres_virtual *
224	    (var->bits_per_pixel >> 3) > info->fix.smem_len)
225		return -EINVAL;
226
227	return 0;
228}
229
230/*
231 * The hardware clock divider has an integer and a fractional
232 * stage:
233 *
234 *	clk2 = clk_in / integer_divider
235 *	clk_out = clk2 * (1 - (fractional_divider >> 12))
236 *
237 * Calculate integer and fractional divider for given clk_in
238 * and clk_out.
239 */
240static void set_clock_divider(struct pxa168fb_info *fbi,
241			      const struct fb_videomode *m)
242{
243	int divider_int;
244	int needed_pixclk;
245	u64 div_result;
246	u32 x = 0;
247
248	/*
249	 * Notice: The field pixclock is used by linux fb
250	 * is in pixel second. E.g. struct fb_videomode &
251	 * struct fb_var_screeninfo
252	 */
253
254	/*
255	 * Check input values.
256	 */
257	if (!m || !m->pixclock || !m->refresh) {
258		dev_err(fbi->dev, "Input refresh or pixclock is wrong.\n");
259		return;
260	}
261
262	/*
263	 * Using PLL/AXI clock.
264	 */
265	x = 0x80000000;
266
267	/*
268	 * Calc divider according to refresh rate.
269	 */
270	div_result = 1000000000000ll;
271	do_div(div_result, m->pixclock);
272	needed_pixclk = (u32)div_result;
273
274	divider_int = clk_get_rate(fbi->clk) / needed_pixclk;
275
276	/* check whether divisor is too small. */
277	if (divider_int < 2) {
278		dev_warn(fbi->dev, "Warning: clock source is too slow. "
279				"Try smaller resolution\n");
280		divider_int = 2;
281	}
282
283	/*
284	 * Set setting to reg.
285	 */
286	x |= divider_int;
287	writel(x, fbi->reg_base + LCD_CFG_SCLK_DIV);
288}
289
290static void set_dma_control0(struct pxa168fb_info *fbi)
291{
292	u32 x;
293
294	/*
295	 * Set bit to enable graphics DMA.
296	 */
297	x = readl(fbi->reg_base + LCD_SPU_DMA_CTRL0);
298	x &= ~CFG_GRA_ENA_MASK;
299	x |= fbi->active ? CFG_GRA_ENA(1) : CFG_GRA_ENA(0);
300
301	/*
302	 * If we are in a pseudo-color mode, we need to enable
303	 * palette lookup.
304	 */
305	if (fbi->pix_fmt == PIX_FMT_PSEUDOCOLOR)
306		x |= 0x10000000;
307
308	/*
309	 * Configure hardware pixel format.
310	 */
311	x &= ~(0xF << 16);
312	x |= (fbi->pix_fmt >> 1) << 16;
313
314	/*
315	 * Check red and blue pixel swap.
316	 * 1. source data swap
317	 * 2. panel output data swap
318	 */
319	x &= ~(1 << 12);
320	x |= ((fbi->pix_fmt & 1) ^ (fbi->panel_rbswap)) << 12;
321
322	writel(x, fbi->reg_base + LCD_SPU_DMA_CTRL0);
323}
324
325static void set_dma_control1(struct pxa168fb_info *fbi, int sync)
326{
327	u32 x;
328
329	/*
330	 * Configure default bits: vsync triggers DMA, gated clock
331	 * enable, power save enable, configure alpha registers to
332	 * display 100% graphics, and set pixel command.
333	 */
334	x = readl(fbi->reg_base + LCD_SPU_DMA_CTRL1);
335	x |= 0x2032ff81;
336
337	/*
338	 * We trigger DMA on the falling edge of vsync if vsync is
339	 * active low, or on the rising edge if vsync is active high.
340	 */
341	if (!(sync & FB_SYNC_VERT_HIGH_ACT))
342		x |= 0x08000000;
343
344	writel(x, fbi->reg_base + LCD_SPU_DMA_CTRL1);
345}
346
347static void set_graphics_start(struct fb_info *info, int xoffset, int yoffset)
348{
349	struct pxa168fb_info *fbi = info->par;
350	struct fb_var_screeninfo *var = &info->var;
351	int pixel_offset;
352	unsigned long addr;
353
354	pixel_offset = (yoffset * var->xres_virtual) + xoffset;
355
356	addr = fbi->fb_start_dma + (pixel_offset * (var->bits_per_pixel >> 3));
357	writel(addr, fbi->reg_base + LCD_CFG_GRA_START_ADDR0);
358}
359
360static void set_dumb_panel_control(struct fb_info *info)
361{
362	struct pxa168fb_info *fbi = info->par;
363	struct pxa168fb_mach_info *mi = dev_get_platdata(fbi->dev);
364	u32 x;
365
366	/*
367	 * Preserve enable flag.
368	 */
369	x = readl(fbi->reg_base + LCD_SPU_DUMB_CTRL) & 0x00000001;
370
371	x |= (fbi->is_blanked ? 0x7 : mi->dumb_mode) << 28;
372	x |= mi->gpio_output_data << 20;
373	x |= mi->gpio_output_mask << 12;
374	x |= mi->panel_rgb_reverse_lanes ? 0x00000080 : 0;
375	x |= mi->invert_composite_blank ? 0x00000040 : 0;
376	x |= (info->var.sync & FB_SYNC_COMP_HIGH_ACT) ? 0x00000020 : 0;
377	x |= mi->invert_pix_val_ena ? 0x00000010 : 0;
378	x |= (info->var.sync & FB_SYNC_VERT_HIGH_ACT) ? 0 : 0x00000008;
379	x |= (info->var.sync & FB_SYNC_HOR_HIGH_ACT) ? 0 : 0x00000004;
380	x |= mi->invert_pixclock ? 0x00000002 : 0;
381
382	writel(x, fbi->reg_base + LCD_SPU_DUMB_CTRL);
383}
384
385static void set_dumb_screen_dimensions(struct fb_info *info)
386{
387	struct pxa168fb_info *fbi = info->par;
388	struct fb_var_screeninfo *v = &info->var;
389	int x;
390	int y;
391
392	x = v->xres + v->right_margin + v->hsync_len + v->left_margin;
393	y = v->yres + v->lower_margin + v->vsync_len + v->upper_margin;
394
395	writel((y << 16) | x, fbi->reg_base + LCD_SPUT_V_H_TOTAL);
396}
397
398static int pxa168fb_set_par(struct fb_info *info)
399{
400	struct pxa168fb_info *fbi = info->par;
401	struct fb_var_screeninfo *var = &info->var;
402	struct fb_videomode mode;
403	u32 x;
404
405	/*
406	 * Set additional mode info.
407	 */
408	if (fbi->pix_fmt == PIX_FMT_PSEUDOCOLOR)
409		info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
410	else
411		info->fix.visual = FB_VISUAL_TRUECOLOR;
412	info->fix.line_length = var->xres_virtual * var->bits_per_pixel / 8;
413	info->fix.ypanstep = var->yres;
414
415	/*
416	 * Disable panel output while we setup the display.
417	 */
418	x = readl(fbi->reg_base + LCD_SPU_DUMB_CTRL);
419	writel(x & ~1, fbi->reg_base + LCD_SPU_DUMB_CTRL);
420
421	/*
422	 * Configure global panel parameters.
423	 */
424	writel((var->yres << 16) | var->xres,
425		fbi->reg_base + LCD_SPU_V_H_ACTIVE);
426
427	/*
428	 * convet var to video mode
429	 */
430	fb_var_to_videomode(&mode, &info->var);
431
432	/* Calculate clock divisor. */
433	set_clock_divider(fbi, &mode);
434
435	/* Configure dma ctrl regs. */
436	set_dma_control0(fbi);
437	set_dma_control1(fbi, info->var.sync);
438
439	/*
440	 * Configure graphics DMA parameters.
441	 */
442	x = readl(fbi->reg_base + LCD_CFG_GRA_PITCH);
443	x = (x & ~0xFFFF) | ((var->xres_virtual * var->bits_per_pixel) >> 3);
444	writel(x, fbi->reg_base + LCD_CFG_GRA_PITCH);
445	writel((var->yres << 16) | var->xres,
446		fbi->reg_base + LCD_SPU_GRA_HPXL_VLN);
447	writel((var->yres << 16) | var->xres,
448		fbi->reg_base + LCD_SPU_GZM_HPXL_VLN);
449
450	/*
451	 * Configure dumb panel ctrl regs & timings.
452	 */
453	set_dumb_panel_control(info);
454	set_dumb_screen_dimensions(info);
455
456	writel((var->left_margin << 16) | var->right_margin,
457			fbi->reg_base + LCD_SPU_H_PORCH);
458	writel((var->upper_margin << 16) | var->lower_margin,
459			fbi->reg_base + LCD_SPU_V_PORCH);
460
461	/*
462	 * Re-enable panel output.
463	 */
464	x = readl(fbi->reg_base + LCD_SPU_DUMB_CTRL);
465	writel(x | 1, fbi->reg_base + LCD_SPU_DUMB_CTRL);
466
467	return 0;
468}
469
470static unsigned int chan_to_field(unsigned int chan, struct fb_bitfield *bf)
471{
472	return ((chan & 0xffff) >> (16 - bf->length)) << bf->offset;
473}
474
475static u32 to_rgb(u16 red, u16 green, u16 blue)
476{
477	red >>= 8;
478	green >>= 8;
479	blue >>= 8;
480
481	return (red << 16) | (green << 8) | blue;
482}
483
484static int
485pxa168fb_setcolreg(unsigned int regno, unsigned int red, unsigned int green,
486		 unsigned int blue, unsigned int trans, struct fb_info *info)
487{
488	struct pxa168fb_info *fbi = info->par;
489	u32 val;
490
491	if (info->var.grayscale)
492		red = green = blue = (19595 * red + 38470 * green +
493					7471 * blue) >> 16;
494
495	if (info->fix.visual == FB_VISUAL_TRUECOLOR && regno < 16) {
496		val =  chan_to_field(red,   &info->var.red);
497		val |= chan_to_field(green, &info->var.green);
498		val |= chan_to_field(blue , &info->var.blue);
499		fbi->pseudo_palette[regno] = val;
500	}
501
502	if (info->fix.visual == FB_VISUAL_PSEUDOCOLOR && regno < 256) {
503		val = to_rgb(red, green, blue);
504		writel(val, fbi->reg_base + LCD_SPU_SRAM_WRDAT);
505		writel(0x8300 | regno, fbi->reg_base + LCD_SPU_SRAM_CTRL);
506	}
507
508	return 0;
509}
510
511static int pxa168fb_blank(int blank, struct fb_info *info)
512{
513	struct pxa168fb_info *fbi = info->par;
514
515	fbi->is_blanked = (blank == FB_BLANK_UNBLANK) ? 0 : 1;
516	set_dumb_panel_control(info);
517
518	return 0;
519}
520
521static int pxa168fb_pan_display(struct fb_var_screeninfo *var,
522				struct fb_info *info)
523{
524	set_graphics_start(info, var->xoffset, var->yoffset);
525
526	return 0;
527}
528
529static irqreturn_t pxa168fb_handle_irq(int irq, void *dev_id)
530{
531	struct pxa168fb_info *fbi = dev_id;
532	u32 isr = readl(fbi->reg_base + SPU_IRQ_ISR);
533
534	if ((isr & GRA_FRAME_IRQ0_ENA_MASK)) {
535
536		writel(isr & (~GRA_FRAME_IRQ0_ENA_MASK),
537			fbi->reg_base + SPU_IRQ_ISR);
538
539		return IRQ_HANDLED;
540	}
541	return IRQ_NONE;
542}
543
544static const struct fb_ops pxa168fb_ops = {
545	.owner		= THIS_MODULE,
546	FB_DEFAULT_IOMEM_OPS,
547	.fb_check_var	= pxa168fb_check_var,
548	.fb_set_par	= pxa168fb_set_par,
549	.fb_setcolreg	= pxa168fb_setcolreg,
550	.fb_blank	= pxa168fb_blank,
551	.fb_pan_display	= pxa168fb_pan_display,
552};
553
554static void pxa168fb_init_mode(struct fb_info *info,
555			      struct pxa168fb_mach_info *mi)
556{
557	struct pxa168fb_info *fbi = info->par;
558	struct fb_var_screeninfo *var = &info->var;
559	u32 total_w, total_h, refresh;
560	u64 div_result;
561	const struct fb_videomode *m;
562
563	/*
564	 * Set default value
565	 */
566	refresh = DEFAULT_REFRESH;
567
568	/* try to find best video mode. */
569	m = fb_find_best_mode(&info->var, &info->modelist);
570	if (m)
571		fb_videomode_to_var(&info->var, m);
572
573	/* Init settings. */
574	var->xres_virtual = var->xres;
575	var->yres_virtual = info->fix.smem_len /
576		(var->xres_virtual * (var->bits_per_pixel >> 3));
577	dev_dbg(fbi->dev, "pxa168fb: find best mode: res = %dx%d\n",
578				var->xres, var->yres);
579
580	/* correct pixclock. */
581	total_w = var->xres + var->left_margin + var->right_margin +
582		  var->hsync_len;
583	total_h = var->yres + var->upper_margin + var->lower_margin +
584		  var->vsync_len;
585
586	div_result = 1000000000000ll;
587	do_div(div_result, total_w * total_h * refresh);
588	var->pixclock = (u32)div_result;
589}
590
591static int pxa168fb_probe(struct platform_device *pdev)
592{
593	struct pxa168fb_mach_info *mi;
594	struct fb_info *info = NULL;
595	struct pxa168fb_info *fbi = NULL;
596	struct resource *res;
597	struct clk *clk;
598	int irq, ret;
599
600	mi = dev_get_platdata(&pdev->dev);
601	if (mi == NULL) {
602		dev_err(&pdev->dev, "no platform data defined\n");
603		return -EINVAL;
604	}
605
606	clk = devm_clk_get(&pdev->dev, "LCDCLK");
607	if (IS_ERR(clk))
608		return dev_err_probe(&pdev->dev, PTR_ERR(clk),
609				     "unable to get LCDCLK");
610
611	res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
612	if (res == NULL) {
613		dev_err(&pdev->dev, "no IO memory defined\n");
614		return -ENOENT;
615	}
616
617	irq = platform_get_irq(pdev, 0);
618	if (irq < 0)
619		return -ENOENT;
620
621	info = framebuffer_alloc(sizeof(struct pxa168fb_info), &pdev->dev);
622	if (info == NULL) {
623		return -ENOMEM;
624	}
625
626	/* Initialize private data */
627	fbi = info->par;
628	fbi->info = info;
629	fbi->clk = clk;
630	fbi->dev = &pdev->dev;
631	fbi->panel_rbswap = mi->panel_rbswap;
632	fbi->is_blanked = 0;
633	fbi->active = mi->active;
634
635	/*
636	 * Initialise static fb parameters.
637	 */
638	info->flags = FBINFO_PARTIAL_PAN_OK |
639		      FBINFO_HWACCEL_XPAN | FBINFO_HWACCEL_YPAN;
640	info->node = -1;
641	strscpy(info->fix.id, mi->id, 16);
642	info->fix.type = FB_TYPE_PACKED_PIXELS;
643	info->fix.type_aux = 0;
644	info->fix.xpanstep = 0;
645	info->fix.ypanstep = 0;
646	info->fix.ywrapstep = 0;
647	info->fix.mmio_start = res->start;
648	info->fix.mmio_len = resource_size(res);
649	info->fix.accel = FB_ACCEL_NONE;
650	info->fbops = &pxa168fb_ops;
651	info->pseudo_palette = fbi->pseudo_palette;
652
653	/*
654	 * Map LCD controller registers.
655	 */
656	fbi->reg_base = devm_ioremap(&pdev->dev, res->start,
657					     resource_size(res));
658	if (fbi->reg_base == NULL) {
659		ret = -ENOMEM;
660		goto failed_free_info;
661	}
662
663	/*
664	 * Allocate framebuffer memory.
665	 */
666	info->fix.smem_len = PAGE_ALIGN(DEFAULT_FB_SIZE);
667
668	info->screen_base = dma_alloc_wc(fbi->dev, info->fix.smem_len,
669					 &fbi->fb_start_dma, GFP_KERNEL);
670	if (info->screen_base == NULL) {
671		ret = -ENOMEM;
672		goto failed_free_info;
673	}
674
675	info->fix.smem_start = (unsigned long)fbi->fb_start_dma;
676	set_graphics_start(info, 0, 0);
677
678	/*
679	 * Set video mode according to platform data.
680	 */
681	set_mode(fbi, &info->var, mi->modes, mi->pix_fmt, 1);
682
683	fb_videomode_to_modelist(mi->modes, mi->num_modes, &info->modelist);
684
685	/*
686	 * init video mode data.
687	 */
688	pxa168fb_init_mode(info, mi);
689
690	/*
691	 * Fill in sane defaults.
692	 */
693	ret = pxa168fb_check_var(&info->var, info);
694	if (ret)
695		goto failed_free_fbmem;
696
697	/*
698	 * enable controller clock
699	 */
700	clk_prepare_enable(fbi->clk);
701
702	pxa168fb_set_par(info);
703
704	/*
705	 * Configure default register values.
706	 */
707	writel(0, fbi->reg_base + LCD_SPU_BLANKCOLOR);
708	writel(mi->io_pin_allocation_mode, fbi->reg_base + SPU_IOPAD_CONTROL);
709	writel(0, fbi->reg_base + LCD_CFG_GRA_START_ADDR1);
710	writel(0, fbi->reg_base + LCD_SPU_GRA_OVSA_HPXL_VLN);
711	writel(0, fbi->reg_base + LCD_SPU_SRAM_PARA0);
712	writel(CFG_CSB_256x32(0x1)|CFG_CSB_256x24(0x1)|CFG_CSB_256x8(0x1),
713		fbi->reg_base + LCD_SPU_SRAM_PARA1);
714
715	/*
716	 * Allocate color map.
717	 */
718	if (fb_alloc_cmap(&info->cmap, 256, 0) < 0) {
719		ret = -ENOMEM;
720		goto failed_free_clk;
721	}
722
723	/*
724	 * Register irq handler.
725	 */
726	ret = devm_request_irq(&pdev->dev, irq, pxa168fb_handle_irq,
727			       IRQF_SHARED, info->fix.id, fbi);
728	if (ret < 0) {
729		dev_err(&pdev->dev, "unable to request IRQ\n");
730		ret = -ENXIO;
731		goto failed_free_cmap;
732	}
733
734	/*
735	 * Enable GFX interrupt
736	 */
737	writel(GRA_FRAME_IRQ0_ENA(0x1), fbi->reg_base + SPU_IRQ_ENA);
738
739	/*
740	 * Register framebuffer.
741	 */
742	ret = register_framebuffer(info);
743	if (ret < 0) {
744		dev_err(&pdev->dev, "Failed to register pxa168-fb: %d\n", ret);
745		ret = -ENXIO;
746		goto failed_free_cmap;
747	}
748
749	platform_set_drvdata(pdev, fbi);
750	return 0;
751
752failed_free_cmap:
753	fb_dealloc_cmap(&info->cmap);
754failed_free_clk:
755	clk_disable_unprepare(fbi->clk);
756failed_free_fbmem:
757	dma_free_wc(fbi->dev, info->fix.smem_len,
758		    info->screen_base, fbi->fb_start_dma);
759failed_free_info:
760	framebuffer_release(info);
761
762	dev_err(&pdev->dev, "frame buffer device init failed with %d\n", ret);
763	return ret;
764}
765
766static void pxa168fb_remove(struct platform_device *pdev)
767{
768	struct pxa168fb_info *fbi = platform_get_drvdata(pdev);
769	struct fb_info *info;
770	unsigned int data;
771
772	if (!fbi)
773		return;
774
775	/* disable DMA transfer */
776	data = readl(fbi->reg_base + LCD_SPU_DMA_CTRL0);
777	data &= ~CFG_GRA_ENA_MASK;
778	writel(data, fbi->reg_base + LCD_SPU_DMA_CTRL0);
779
780	info = fbi->info;
781
782	unregister_framebuffer(info);
783
784	writel(GRA_FRAME_IRQ0_ENA(0x0), fbi->reg_base + SPU_IRQ_ENA);
785
786	if (info->cmap.len)
787		fb_dealloc_cmap(&info->cmap);
788
789	dma_free_wc(fbi->dev, info->fix.smem_len,
790		    info->screen_base, info->fix.smem_start);
791
792	clk_disable_unprepare(fbi->clk);
793
794	framebuffer_release(info);
795}
796
797static struct platform_driver pxa168fb_driver = {
798	.driver		= {
799		.name	= "pxa168-fb",
800	},
801	.probe		= pxa168fb_probe,
802	.remove_new	= pxa168fb_remove,
803};
804
805module_platform_driver(pxa168fb_driver);
806
807MODULE_AUTHOR("Lennert Buytenhek <buytenh@marvell.com> "
808	      "Green Wan <gwan@marvell.com>");
809MODULE_DESCRIPTION("Framebuffer driver for PXA168/910");
810MODULE_LICENSE("GPL");
811