Blame view

drivers/video/cg14.c 14.7 KB
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
1
2
  /* cg14.c: CGFOURTEEN frame buffer driver
   *
50312ce9d   David S. Miller   [SPARC]: Convert ...
3
   * Copyright (C) 2003, 2006 David S. Miller (davem@davemloft.net)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
4
5
6
7
8
9
10
11
12
13
   * Copyright (C) 1996,1998 Jakub Jelinek (jj@ultra.linux.cz)
   * Copyright (C) 1995 Miguel de Icaza (miguel@nuclecu.unam.mx)
   *
   * Driver layout based loosely on tgafb.c, see that file for credits.
   */
  
  #include <linux/module.h>
  #include <linux/kernel.h>
  #include <linux/errno.h>
  #include <linux/string.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
14
15
16
17
  #include <linux/delay.h>
  #include <linux/init.h>
  #include <linux/fb.h>
  #include <linux/mm.h>
631a9dca6   Robert Reif   [SPARC]: video/cg...
18
  #include <linux/uaccess.h>
6cd5a86b5   Robert Reif   sparc video: remo...
19
  #include <linux/of_device.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
20
21
  
  #include <asm/io.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
22
23
24
25
26
27
28
29
30
31
  #include <asm/fbio.h>
  
  #include "sbuslib.h"
  
  /*
   * Local functions.
   */
  
  static int cg14_setcolreg(unsigned, unsigned, unsigned, unsigned,
  			 unsigned, struct fb_info *);
216d526c8   Christoph Hellwig   [PATCH] fbdev: Sa...
32
  static int cg14_mmap(struct fb_info *, struct vm_area_struct *);
67a6680d6   Christoph Hellwig   [PATCH] fbdev: Sa...
33
  static int cg14_ioctl(struct fb_info *, unsigned int, unsigned long);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
  static int cg14_pan_display(struct fb_var_screeninfo *, struct fb_info *);
  
  /*
   *  Frame buffer operations
   */
  
  static struct fb_ops cg14_ops = {
  	.owner			= THIS_MODULE,
  	.fb_setcolreg		= cg14_setcolreg,
  	.fb_pan_display		= cg14_pan_display,
  	.fb_fillrect		= cfb_fillrect,
  	.fb_copyarea		= cfb_copyarea,
  	.fb_imageblit		= cfb_imageblit,
  	.fb_mmap		= cg14_mmap,
  	.fb_ioctl		= cg14_ioctl,
9ffb83bcc   Christoph Hellwig   [SBUSFB]: impleme...
49
50
51
  #ifdef CONFIG_COMPAT
  	.fb_compat_ioctl	= sbusfb_compat_ioctl,
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
52
53
54
55
56
57
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
  };
  
  #define CG14_MCR_INTENABLE_SHIFT	7
  #define CG14_MCR_INTENABLE_MASK		0x80
  #define CG14_MCR_VIDENABLE_SHIFT	6
  #define CG14_MCR_VIDENABLE_MASK		0x40
  #define CG14_MCR_PIXMODE_SHIFT		4
  #define CG14_MCR_PIXMODE_MASK		0x30
  #define CG14_MCR_TMR_SHIFT		2
  #define CG14_MCR_TMR_MASK		0x0c
  #define CG14_MCR_TMENABLE_SHIFT		1
  #define CG14_MCR_TMENABLE_MASK		0x02
  #define CG14_MCR_RESET_SHIFT		0
  #define CG14_MCR_RESET_MASK		0x01
  #define CG14_REV_REVISION_SHIFT		4
  #define CG14_REV_REVISION_MASK		0xf0
  #define CG14_REV_IMPL_SHIFT		0
  #define CG14_REV_IMPL_MASK		0x0f
  #define CG14_VBR_FRAMEBASE_SHIFT	12
  #define CG14_VBR_FRAMEBASE_MASK		0x00fff000
  #define CG14_VMCR1_SETUP_SHIFT		0
  #define CG14_VMCR1_SETUP_MASK		0x000001ff
  #define CG14_VMCR1_VCONFIG_SHIFT	9
  #define CG14_VMCR1_VCONFIG_MASK		0x00000e00
  #define CG14_VMCR2_REFRESH_SHIFT	0
  #define CG14_VMCR2_REFRESH_MASK		0x00000001
  #define CG14_VMCR2_TESTROWCNT_SHIFT	1
  #define CG14_VMCR2_TESTROWCNT_MASK	0x00000002
  #define CG14_VMCR2_FBCONFIG_SHIFT	2
  #define CG14_VMCR2_FBCONFIG_MASK	0x0000000c
  #define CG14_VCR_REFRESHREQ_SHIFT	0
  #define CG14_VCR_REFRESHREQ_MASK	0x000003ff
  #define CG14_VCR1_REFRESHENA_SHIFT	10
  #define CG14_VCR1_REFRESHENA_MASK	0x00000400
  #define CG14_VCA_CAD_SHIFT		0
  #define CG14_VCA_CAD_MASK		0x000003ff
  #define CG14_VCA_VERS_SHIFT		10
  #define CG14_VCA_VERS_MASK		0x00000c00
  #define CG14_VCA_RAMSPEED_SHIFT		12
  #define CG14_VCA_RAMSPEED_MASK		0x00001000
  #define CG14_VCA_8MB_SHIFT		13
  #define CG14_VCA_8MB_MASK		0x00002000
  
  #define CG14_MCR_PIXMODE_8		0
  #define CG14_MCR_PIXMODE_16		2
  #define CG14_MCR_PIXMODE_32		3
  
  struct cg14_regs{
50312ce9d   David S. Miller   [SPARC]: Convert ...
100
101
102
103
104
105
106
107
108
109
  	u8 mcr;	/* Master Control Reg */
  	u8 ppr;	/* Packed Pixel Reg */
  	u8 tms[2];	/* Test Mode Status Regs */
  	u8 msr;	/* Master Status Reg */
  	u8 fsr;	/* Fault Status Reg */
  	u8 rev;	/* Revision & Impl */
  	u8 ccr;	/* Clock Control Reg */
  	u32 tmr;	/* Test Mode Read Back */
  	u8 mod;	/* Monitor Operation Data Reg */
  	u8 acr;	/* Aux Control */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
110
  	u8 xxx0[6];
50312ce9d   David S. Miller   [SPARC]: Convert ...
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
  	u16 hct;	/* Hor Counter */
  	u16 vct;	/* Vert Counter */
  	u16 hbs;	/* Hor Blank Start */
  	u16 hbc;	/* Hor Blank Clear */
  	u16 hss;	/* Hor Sync Start */
  	u16 hsc;	/* Hor Sync Clear */
  	u16 csc;	/* Composite Sync Clear */
  	u16 vbs;	/* Vert Blank Start */
  	u16 vbc;	/* Vert Blank Clear */
  	u16 vss;	/* Vert Sync Start */
  	u16 vsc;	/* Vert Sync Clear */
  	u16 xcs;
  	u16 xcc;
  	u16 fsa;	/* Fault Status Address */
  	u16 adr;	/* Address Registers */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
126
  	u8 xxx1[0xce];
50312ce9d   David S. Miller   [SPARC]: Convert ...
127
128
129
130
131
  	u8 pcg[0x100]; /* Pixel Clock Generator */
  	u32 vbr;	/* Frame Base Row */
  	u32 vmcr;	/* VBC Master Control */
  	u32 vcr;	/* VBC refresh */
  	u32 vca;	/* VBC Config */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
132
133
134
135
136
137
  };
  
  #define CG14_CCR_ENABLE	0x04
  #define CG14_CCR_SELECT 0x02	/* HW/Full screen */
  
  struct cg14_cursor {
50312ce9d   David S. Miller   [SPARC]: Convert ...
138
139
140
  	u32 cpl0[32];	/* Enable plane 0 */
  	u32 cpl1[32];  /* Color selection plane */
  	u8 ccr;	/* Cursor Control Reg */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
141
  	u8 xxx0[3];
50312ce9d   David S. Miller   [SPARC]: Convert ...
142
143
144
145
  	u16 cursx;	/* Cursor x,y position */
  	u16 cursy;	/* Cursor x,y position */
  	u32 color0;
  	u32 color1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
146
  	u32 xxx1[0x1bc];
50312ce9d   David S. Miller   [SPARC]: Convert ...
147
148
  	u32 cpl0i[32];	/* Enable plane 0 autoinc */
  	u32 cpl1i[32]; /* Color selection autoinc */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
149
150
151
  };
  
  struct cg14_dac {
50312ce9d   David S. Miller   [SPARC]: Convert ...
152
  	u8 addr;	/* Address Register */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
153
  	u8 xxx0[255];
50312ce9d   David S. Miller   [SPARC]: Convert ...
154
  	u8 glut;	/* Gamma table */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
155
  	u8 xxx1[255];
50312ce9d   David S. Miller   [SPARC]: Convert ...
156
  	u8 select;	/* Register Select */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
157
  	u8 xxx2[255];
50312ce9d   David S. Miller   [SPARC]: Convert ...
158
  	u8 mode;	/* Mode Register */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
159
160
161
  };
  
  struct cg14_xlut{
50312ce9d   David S. Miller   [SPARC]: Convert ...
162
163
  	u8 x_xlut [256];
  	u8 x_xlutd [256];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
164
  	u8 xxx0[0x600];
50312ce9d   David S. Miller   [SPARC]: Convert ...
165
166
  	u8 x_xlut_inc [256];
  	u8 x_xlutd_inc [256];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
  };
  
  /* Color look up table (clut) */
  /* Each one of these arrays hold the color lookup table (for 256
   * colors) for each MDI page (I assume then there should be 4 MDI
   * pages, I still wonder what they are.  I have seen NeXTStep split
   * the screen in four parts, while operating in 24 bits mode.  Each
   * integer holds 4 values: alpha value (transparency channel, thanks
   * go to John Stone (johns@umr.edu) from OpenBSD), red, green and blue
   *
   * I currently use the clut instead of the Xlut
   */
  struct cg14_clut {
  	u32 c_clut [256];
  	u32 c_clutd [256];    /* i wonder what the 'd' is for */
  	u32 c_clut_inc [256];
  	u32 c_clutd_inc [256];
  };
  
  #define CG14_MMAP_ENTRIES	16
  
  struct cg14_par {
  	spinlock_t		lock;
  	struct cg14_regs	__iomem *regs;
  	struct cg14_clut	__iomem *clut;
  	struct cg14_cursor	__iomem *cursor;
  
  	u32			flags;
  #define CG14_FLAG_BLANKED	0x00000001
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
196
  	unsigned long		iospace;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
197
198
199
200
201
  
  	struct sbus_mmap_map	mmap_map[CG14_MMAP_ENTRIES];
  
  	int			mode;
  	int			ramsize;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
  };
  
  static void __cg14_reset(struct cg14_par *par)
  {
  	struct cg14_regs __iomem *regs = par->regs;
  	u8 val;
  
  	val = sbus_readb(&regs->mcr);
  	val &= ~(CG14_MCR_PIXMODE_MASK);
  	sbus_writeb(val, &regs->mcr);
  }
  
  static int cg14_pan_display(struct fb_var_screeninfo *var, struct fb_info *info)
  {
  	struct cg14_par *par = (struct cg14_par *) info->par;
  	unsigned long flags;
  
  	/* We just use this to catch switches out of
  	 * graphics mode.
  	 */
  	spin_lock_irqsave(&par->lock, flags);
  	__cg14_reset(par);
  	spin_unlock_irqrestore(&par->lock, flags);
  
  	if (var->xoffset || var->yoffset || var->vmode)
  		return -EINVAL;
  	return 0;
  }
  
  /**
   *      cg14_setcolreg - Optional function. Sets a color register.
   *      @regno: boolean, 0 copy local, 1 get_user() function
   *      @red: frame buffer colormap structure
   *      @green: The green value which can be up to 16 bits wide
   *      @blue:  The blue value which can be up to 16 bits wide.
   *      @transp: If supported the alpha value which can be up to 16 bits wide.
   *      @info: frame buffer info structure
   */
  static int cg14_setcolreg(unsigned regno,
  			  unsigned red, unsigned green, unsigned blue,
  			  unsigned transp, struct fb_info *info)
  {
  	struct cg14_par *par = (struct cg14_par *) info->par;
  	struct cg14_clut __iomem *clut = par->clut;
  	unsigned long flags;
  	u32 val;
  
  	if (regno >= 256)
  		return 1;
  
  	red >>= 8;
  	green >>= 8;
  	blue >>= 8;
  	val = (red | (green << 8) | (blue << 16));
  
  	spin_lock_irqsave(&par->lock, flags);
  	sbus_writel(val, &clut->c_clut[regno]);
  	spin_unlock_irqrestore(&par->lock, flags);
  
  	return 0;
  }
216d526c8   Christoph Hellwig   [PATCH] fbdev: Sa...
263
  static int cg14_mmap(struct fb_info *info, struct vm_area_struct *vma)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
264
265
266
267
  {
  	struct cg14_par *par = (struct cg14_par *) info->par;
  
  	return sbusfb_mmap_helper(par->mmap_map,
de4dc48ee   Krzysztof Helt   cg14: use standar...
268
  				  info->fix.smem_start, info->fix.smem_len,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
269
270
  				  par->iospace, vma);
  }
67a6680d6   Christoph Hellwig   [PATCH] fbdev: Sa...
271
  static int cg14_ioctl(struct fb_info *info, unsigned int cmd, unsigned long arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
  {
  	struct cg14_par *par = (struct cg14_par *) info->par;
  	struct cg14_regs __iomem *regs = par->regs;
  	struct mdi_cfginfo kmdi, __user *mdii;
  	unsigned long flags;
  	int cur_mode, mode, ret = 0;
  
  	switch (cmd) {
  	case MDI_RESET:
  		spin_lock_irqsave(&par->lock, flags);
  		__cg14_reset(par);
  		spin_unlock_irqrestore(&par->lock, flags);
  		break;
  
  	case MDI_GET_CFGINFO:
  		memset(&kmdi, 0, sizeof(kmdi));
  
  		spin_lock_irqsave(&par->lock, flags);
  		kmdi.mdi_type = FBTYPE_MDICOLOR;
  		kmdi.mdi_height = info->var.yres;
  		kmdi.mdi_width = info->var.xres;
  		kmdi.mdi_mode = par->mode;
  		kmdi.mdi_pixfreq = 72; /* FIXME */
  		kmdi.mdi_size = par->ramsize;
  		spin_unlock_irqrestore(&par->lock, flags);
  
  		mdii = (struct mdi_cfginfo __user *) arg;
  		if (copy_to_user(mdii, &kmdi, sizeof(kmdi)))
  			ret = -EFAULT;
  		break;
  
  	case MDI_SET_PIXELMODE:
  		if (get_user(mode, (int __user *) arg)) {
  			ret = -EFAULT;
  			break;
  		}
  
  		spin_lock_irqsave(&par->lock, flags);
  		cur_mode = sbus_readb(&regs->mcr);
  		cur_mode &= ~CG14_MCR_PIXMODE_MASK;
  		switch(mode) {
  		case MDI_32_PIX:
  			cur_mode |= (CG14_MCR_PIXMODE_32 <<
  				     CG14_MCR_PIXMODE_SHIFT);
  			break;
  
  		case MDI_16_PIX:
  			cur_mode |= (CG14_MCR_PIXMODE_16 <<
  				     CG14_MCR_PIXMODE_SHIFT);
  			break;
  
  		case MDI_8_PIX:
  			break;
  
  		default:
  			ret = -ENOSYS;
  			break;
  		};
  		if (!ret) {
  			sbus_writeb(cur_mode, &regs->mcr);
  			par->mode = mode;
  		}
  		spin_unlock_irqrestore(&par->lock, flags);
  		break;
  
  	default:
  		ret = sbusfb_ioctl_helper(cmd, arg, info,
de4dc48ee   Krzysztof Helt   cg14: use standar...
339
340
  					  FBTYPE_MDICOLOR, 8,
  					  info->fix.smem_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
341
342
343
344
345
346
347
348
349
  		break;
  	};
  
  	return ret;
  }
  
  /*
   *  Initialisation
   */
8fdab4763   Robert Reif   [CG14]: Fix secti...
350
351
  static void __devinit cg14_init_fix(struct fb_info *info, int linebytes,
  				    struct device_node *dp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
352
  {
50312ce9d   David S. Miller   [SPARC]: Convert ...
353
  	const char *name = dp->name;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
354
355
356
357
358
359
360
361
362
363
  
  	strlcpy(info->fix.id, name, sizeof(info->fix.id));
  
  	info->fix.type = FB_TYPE_PACKED_PIXELS;
  	info->fix.visual = FB_VISUAL_PSEUDOCOLOR;
  
  	info->fix.line_length = linebytes;
  
  	info->fix.accel = FB_ACCEL_SUN_CG14;
  }
8fdab4763   Robert Reif   [CG14]: Fix secti...
364
  static struct sbus_mmap_map __cg14_mmap_map[CG14_MMAP_ENTRIES] __devinitdata = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
  	{
  		.voff	= CG14_REGS,
  		.poff	= 0x80000000,
  		.size	= 0x1000
  	},
  	{
  		.voff	= CG14_XLUT,
  		.poff	= 0x80003000,
  		.size	= 0x1000
  	},
  	{
  		.voff	= CG14_CLUT1,
  		.poff	= 0x80004000,
  		.size	= 0x1000
  	},
  	{
  		.voff	= CG14_CLUT2,
  		.poff	= 0x80005000,
  		.size	= 0x1000
  	},
  	{
  		.voff	= CG14_CLUT3,
  		.poff	= 0x80006000,
  		.size	= 0x1000
  	},
  	{
  		.voff	= CG3_MMAP_OFFSET - 0x7000,
  		.poff	= 0x80000000,
  		.size	= 0x7000
  	},
  	{
  		.voff	= CG3_MMAP_OFFSET,
  		.poff	= 0x00000000,
  		.size	= SBUS_MMAP_FBSIZE(1)
  	},
  	{
  		.voff	= MDI_CURSOR_MAP,
  		.poff	= 0x80001000,
  		.size	= 0x1000
  	},
  	{
  		.voff	= MDI_CHUNKY_BGR_MAP,
  		.poff	= 0x01000000,
  		.size	= 0x400000
  	},
  	{
  		.voff	= MDI_PLANAR_X16_MAP,
  		.poff	= 0x02000000,
  		.size	= 0x200000
  	},
  	{
  		.voff	= MDI_PLANAR_C16_MAP,
  		.poff	= 0x02800000,
  		.size	= 0x200000
  	},
  	{
  		.voff	= MDI_PLANAR_X32_MAP,
  		.poff	= 0x03000000,
  		.size	= 0x100000
  	},
  	{
  		.voff	= MDI_PLANAR_B32_MAP,
  		.poff	= 0x03400000,
  		.size	= 0x100000
  	},
  	{
  		.voff	= MDI_PLANAR_G32_MAP,
  		.poff	= 0x03800000,
  		.size	= 0x100000
  	},
  	{
  		.voff	= MDI_PLANAR_R32_MAP,
  		.poff	= 0x03c00000,
  		.size	= 0x100000
  	},
  	{ .size = 0 }
  };
2dc115813   Grant Likely   of/device: Replac...
442
  static void cg14_unmap_regs(struct platform_device *op, struct fb_info *info,
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
443
  			    struct cg14_par *par)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
444
  {
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
445
  	if (par->regs)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
446
  		of_iounmap(&op->resource[0],
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
447
448
  			   par->regs, sizeof(struct cg14_regs));
  	if (par->clut)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
449
  		of_iounmap(&op->resource[0],
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
450
451
  			   par->clut, sizeof(struct cg14_clut));
  	if (par->cursor)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
452
  		of_iounmap(&op->resource[0],
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
453
454
  			   par->cursor, sizeof(struct cg14_cursor));
  	if (info->screen_base)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
455
  		of_iounmap(&op->resource[1],
de4dc48ee   Krzysztof Helt   cg14: use standar...
456
  			   info->screen_base, info->fix.smem_len);
50312ce9d   David S. Miller   [SPARC]: Convert ...
457
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
458

28541d0f1   Grant Likely   dt/video: Elimina...
459
  static int __devinit cg14_probe(struct platform_device *op)
50312ce9d   David S. Miller   [SPARC]: Convert ...
460
  {
d4b8b2c2c   Anatolij Gustschin   of/video: fix bui...
461
  	struct device_node *dp = op->dev.of_node;
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
462
463
  	struct fb_info *info;
  	struct cg14_par *par;
50312ce9d   David S. Miller   [SPARC]: Convert ...
464
  	int is_8mb, linebytes, i, err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
465

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
466
467
468
469
470
471
  	info = framebuffer_alloc(sizeof(struct cg14_par), &op->dev);
  
  	err = -ENOMEM;
  	if (!info)
  		goto out_err;
  	par = info->par;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
472

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
473
  	spin_lock_init(&par->lock);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
474

6cd5a86b5   Robert Reif   sparc video: remo...
475
  	sbusfb_fill_var(&info->var, dp, 8);
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
476
477
478
  	info->var.red.length = 8;
  	info->var.green.length = 8;
  	info->var.blue.length = 8;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
479

50312ce9d   David S. Miller   [SPARC]: Convert ...
480
  	linebytes = of_getintprop_default(dp, "linebytes",
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
481
  					  info->var.xres);
de4dc48ee   Krzysztof Helt   cg14: use standar...
482
  	info->fix.smem_len = PAGE_ALIGN(linebytes * info->var.yres);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
483

50312ce9d   David S. Miller   [SPARC]: Convert ...
484
485
  	if (!strcmp(dp->parent->name, "sbus") ||
  	    !strcmp(dp->parent->name, "sbi")) {
de4dc48ee   Krzysztof Helt   cg14: use standar...
486
  		info->fix.smem_start = op->resource[0].start;
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
487
  		par->iospace = op->resource[0].flags & IORESOURCE_BITS;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
488
  	} else {
de4dc48ee   Krzysztof Helt   cg14: use standar...
489
  		info->fix.smem_start = op->resource[1].start;
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
490
  		par->iospace = op->resource[0].flags & IORESOURCE_BITS;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
491
  	}
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
492
493
494
495
496
497
  	par->regs = of_ioremap(&op->resource[0], 0,
  			       sizeof(struct cg14_regs), "cg14 regs");
  	par->clut = of_ioremap(&op->resource[0], CG14_CLUT1,
  			       sizeof(struct cg14_clut), "cg14 clut");
  	par->cursor = of_ioremap(&op->resource[0], CG14_CURSORREGS,
  				 sizeof(struct cg14_cursor), "cg14 cursor");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
498

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
499
  	info->screen_base = of_ioremap(&op->resource[1], 0,
de4dc48ee   Krzysztof Helt   cg14: use standar...
500
  				       info->fix.smem_len, "cg14 ram");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
501

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
502
503
  	if (!par->regs || !par->clut || !par->cursor || !info->screen_base)
  		goto out_unmap_regs;
50312ce9d   David S. Miller   [SPARC]: Convert ...
504
505
506
  
  	is_8mb = (((op->resource[1].end - op->resource[1].start) + 1) ==
  		  (8 * 1024 * 1024));
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
507
  	BUILD_BUG_ON(sizeof(par->mmap_map) != sizeof(__cg14_mmap_map));
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
508
  		
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
509
  	memcpy(&par->mmap_map, &__cg14_mmap_map, sizeof(par->mmap_map));
50312ce9d   David S. Miller   [SPARC]: Convert ...
510

1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
511
  	for (i = 0; i < CG14_MMAP_ENTRIES; i++) {
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
512
  		struct sbus_mmap_map *map = &par->mmap_map[i];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
513
514
515
516
  
  		if (!map->size)
  			break;
  		if (map->poff & 0x80000000)
50312ce9d   David S. Miller   [SPARC]: Convert ...
517
518
519
  			map->poff = (map->poff & 0x7fffffff) +
  				(op->resource[0].start -
  				 op->resource[1].start);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
520
521
522
523
524
  		if (is_8mb &&
  		    map->size >= 0x100000 &&
  		    map->size <= 0x400000)
  			map->size *= 2;
  	}
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
525
526
  	par->mode = MDI_8_PIX;
  	par->ramsize = (is_8mb ? 0x800000 : 0x400000);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
527

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
528
529
  	info->flags = FBINFO_DEFAULT | FBINFO_HWACCEL_YPAN;
  	info->fbops = &cg14_ops;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
530

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
531
  	__cg14_reset(par);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
532

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
533
534
  	if (fb_alloc_cmap(&info->cmap, 256, 0))
  		goto out_unmap_regs;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
535

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
536
  	fb_set_cmap(&info->cmap, info);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
537

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
538
539
540
541
542
  	cg14_init_fix(info, linebytes, dp);
  
  	err = register_framebuffer(info);
  	if (err < 0)
  		goto out_dealloc_cmap;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
543

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
544
  	dev_set_drvdata(&op->dev, info);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
545

194f1a68b   Robert Reif   sparc: video driv...
546
547
  	printk(KERN_INFO "%s: cgfourteen at %lx:%lx, %dMB
  ",
50312ce9d   David S. Miller   [SPARC]: Convert ...
548
  	       dp->full_name,
de4dc48ee   Krzysztof Helt   cg14: use standar...
549
  	       par->iospace, info->fix.smem_start,
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
550
  	       par->ramsize >> 20);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
551

50312ce9d   David S. Miller   [SPARC]: Convert ...
552
  	return 0;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
553

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
554
555
556
557
558
  out_dealloc_cmap:
  	fb_dealloc_cmap(&info->cmap);
  
  out_unmap_regs:
  	cg14_unmap_regs(op, info, par);
6359691d7   Axel Lin   video: add missin...
559
  	framebuffer_release(info);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
560

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
561
562
  out_err:
  	return err;
50312ce9d   David S. Miller   [SPARC]: Convert ...
563
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
564

2dc115813   Grant Likely   of/device: Replac...
565
  static int __devexit cg14_remove(struct platform_device *op)
50312ce9d   David S. Miller   [SPARC]: Convert ...
566
  {
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
567
568
  	struct fb_info *info = dev_get_drvdata(&op->dev);
  	struct cg14_par *par = info->par;
50312ce9d   David S. Miller   [SPARC]: Convert ...
569

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
570
571
  	unregister_framebuffer(info);
  	fb_dealloc_cmap(&info->cmap);
50312ce9d   David S. Miller   [SPARC]: Convert ...
572

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
573
  	cg14_unmap_regs(op, info, par);
50312ce9d   David S. Miller   [SPARC]: Convert ...
574

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
575
  	framebuffer_release(info);
50312ce9d   David S. Miller   [SPARC]: Convert ...
576

e3a411a3d   David S. Miller   [SPARC64]: Fix of...
577
  	dev_set_drvdata(&op->dev, NULL);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
578
579
580
  
  	return 0;
  }
fd098316e   David S. Miller   sparc: Annotate o...
581
  static const struct of_device_id cg14_match[] = {
50312ce9d   David S. Miller   [SPARC]: Convert ...
582
583
584
585
586
587
  	{
  		.name = "cgfourteen",
  	},
  	{},
  };
  MODULE_DEVICE_TABLE(of, cg14_match);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
588

28541d0f1   Grant Likely   dt/video: Elimina...
589
  static struct platform_driver cg14_driver = {
4018294b5   Grant Likely   of: Remove duplic...
590
591
592
593
594
  	.driver = {
  		.name = "cg14",
  		.owner = THIS_MODULE,
  		.of_match_table = cg14_match,
  	},
50312ce9d   David S. Miller   [SPARC]: Convert ...
595
596
597
  	.probe		= cg14_probe,
  	.remove		= __devexit_p(cg14_remove),
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
598

544330009   Robert Reif   sparc: cg14.c mak...
599
  static int __init cg14_init(void)
50312ce9d   David S. Miller   [SPARC]: Convert ...
600
601
602
  {
  	if (fb_get_options("cg14fb", NULL))
  		return -ENODEV;
28541d0f1   Grant Likely   dt/video: Elimina...
603
  	return platform_driver_register(&cg14_driver);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
604
  }
544330009   Robert Reif   sparc: cg14.c mak...
605
  static void __exit cg14_exit(void)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
606
  {
28541d0f1   Grant Likely   dt/video: Elimina...
607
  	platform_driver_unregister(&cg14_driver);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
608
609
610
  }
  
  module_init(cg14_init);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
611
  module_exit(cg14_exit);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
612
613
  
  MODULE_DESCRIPTION("framebuffer driver for CGfourteen chipsets");
50312ce9d   David S. Miller   [SPARC]: Convert ...
614
615
  MODULE_AUTHOR("David S. Miller <davem@davemloft.net>");
  MODULE_VERSION("2.0");
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
616
  MODULE_LICENSE("GPL");