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
14
15
16
17
18
   * 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>
  #include <linux/slab.h>
  #include <linux/delay.h>
  #include <linux/init.h>
  #include <linux/fb.h>
  #include <linux/mm.h>
631a9dca6   Robert Reif   [SPARC]: video/cg...
19
  #include <linux/uaccess.h>
6cd5a86b5   Robert Reif   sparc video: remo...
20
  #include <linux/of_device.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
21
22
  
  #include <asm/io.h>
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
23
24
25
26
27
28
29
30
31
32
  #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...
33
  static int cg14_mmap(struct fb_info *, struct vm_area_struct *);
67a6680d6   Christoph Hellwig   [PATCH] fbdev: Sa...
34
  static int cg14_ioctl(struct fb_info *, unsigned int, unsigned long);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
  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...
50
51
52
  #ifdef CONFIG_COMPAT
  	.fb_compat_ioctl	= sbusfb_compat_ioctl,
  #endif
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
100
  };
  
  #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 ...
101
102
103
104
105
106
107
108
109
110
  	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
111
  	u8 xxx0[6];
50312ce9d   David S. Miller   [SPARC]: Convert ...
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
  	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
127
  	u8 xxx1[0xce];
50312ce9d   David S. Miller   [SPARC]: Convert ...
128
129
130
131
132
  	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
133
134
135
136
137
138
  };
  
  #define CG14_CCR_ENABLE	0x04
  #define CG14_CCR_SELECT 0x02	/* HW/Full screen */
  
  struct cg14_cursor {
50312ce9d   David S. Miller   [SPARC]: Convert ...
139
140
141
  	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
142
  	u8 xxx0[3];
50312ce9d   David S. Miller   [SPARC]: Convert ...
143
144
145
146
  	u16 cursx;	/* Cursor x,y position */
  	u16 cursy;	/* Cursor x,y position */
  	u32 color0;
  	u32 color1;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
147
  	u32 xxx1[0x1bc];
50312ce9d   David S. Miller   [SPARC]: Convert ...
148
149
  	u32 cpl0i[32];	/* Enable plane 0 autoinc */
  	u32 cpl1i[32]; /* Color selection autoinc */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
150
151
152
  };
  
  struct cg14_dac {
50312ce9d   David S. Miller   [SPARC]: Convert ...
153
  	u8 addr;	/* Address Register */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
154
  	u8 xxx0[255];
50312ce9d   David S. Miller   [SPARC]: Convert ...
155
  	u8 glut;	/* Gamma table */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
156
  	u8 xxx1[255];
50312ce9d   David S. Miller   [SPARC]: Convert ...
157
  	u8 select;	/* Register Select */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
158
  	u8 xxx2[255];
50312ce9d   David S. Miller   [SPARC]: Convert ...
159
  	u8 mode;	/* Mode Register */
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
160
161
162
  };
  
  struct cg14_xlut{
50312ce9d   David S. Miller   [SPARC]: Convert ...
163
164
  	u8 x_xlut [256];
  	u8 x_xlutd [256];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
165
  	u8 xxx0[0x600];
50312ce9d   David S. Miller   [SPARC]: Convert ...
166
167
  	u8 x_xlut_inc [256];
  	u8 x_xlutd_inc [256];
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
196
  };
  
  /* 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
197
  	unsigned long		iospace;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
198
199
200
201
202
  
  	struct sbus_mmap_map	mmap_map[CG14_MMAP_ENTRIES];
  
  	int			mode;
  	int			ramsize;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
263
  };
  
  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...
264
  static int cg14_mmap(struct fb_info *info, struct vm_area_struct *vma)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
265
266
267
268
  {
  	struct cg14_par *par = (struct cg14_par *) info->par;
  
  	return sbusfb_mmap_helper(par->mmap_map,
de4dc48ee   Krzysztof Helt   cg14: use standar...
269
  				  info->fix.smem_start, info->fix.smem_len,
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
270
271
  				  par->iospace, vma);
  }
67a6680d6   Christoph Hellwig   [PATCH] fbdev: Sa...
272
  static int cg14_ioctl(struct fb_info *info, unsigned int cmd, unsigned long arg)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
339
  {
  	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...
340
341
  					  FBTYPE_MDICOLOR, 8,
  					  info->fix.smem_len);
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
342
343
344
345
346
347
348
349
350
  		break;
  	};
  
  	return ret;
  }
  
  /*
   *  Initialisation
   */
8fdab4763   Robert Reif   [CG14]: Fix secti...
351
352
  static void __devinit cg14_init_fix(struct fb_info *info, int linebytes,
  				    struct device_node *dp)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
353
  {
50312ce9d   David S. Miller   [SPARC]: Convert ...
354
  	const char *name = dp->name;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
355
356
357
358
359
360
361
362
363
364
  
  	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...
365
  static struct sbus_mmap_map __cg14_mmap_map[CG14_MMAP_ENTRIES] __devinitdata = {
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
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
442
  	{
  		.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 }
  };
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
443
444
  static void cg14_unmap_regs(struct of_device *op, struct fb_info *info,
  			    struct cg14_par *par)
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
445
  {
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
446
  	if (par->regs)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
447
  		of_iounmap(&op->resource[0],
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
448
449
  			   par->regs, sizeof(struct cg14_regs));
  	if (par->clut)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
450
  		of_iounmap(&op->resource[0],
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
451
452
  			   par->clut, sizeof(struct cg14_clut));
  	if (par->cursor)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
453
  		of_iounmap(&op->resource[0],
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
454
455
  			   par->cursor, sizeof(struct cg14_cursor));
  	if (info->screen_base)
e3a411a3d   David S. Miller   [SPARC64]: Fix of...
456
  		of_iounmap(&op->resource[1],
de4dc48ee   Krzysztof Helt   cg14: use standar...
457
  			   info->screen_base, info->fix.smem_len);
50312ce9d   David S. Miller   [SPARC]: Convert ...
458
  }
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
459

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
460
  static int __devinit cg14_probe(struct of_device *op, const struct of_device_id *match)
50312ce9d   David S. Miller   [SPARC]: Convert ...
461
462
  {
  	struct device_node *dp = op->node;
c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
463
464
  	struct fb_info *info;
  	struct cg14_par *par;
50312ce9d   David S. Miller   [SPARC]: Convert ...
465
  	int is_8mb, linebytes, i, err;
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
466

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
467
468
469
470
471
472
  	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
473

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

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

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

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

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

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

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

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

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

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

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

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
539
540
541
542
543
  	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
544

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

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

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

c7f439b99   David S. Miller   [VIDEO]: Fix OOPS...
555
556
557
558
559
  out_dealloc_cmap:
  	fb_dealloc_cmap(&info->cmap);
  
  out_unmap_regs:
  	cg14_unmap_regs(op, info, par);
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

e3a411a3d   David S. Miller   [SPARC64]: Fix of...
565
  static int __devexit cg14_remove(struct of_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

50312ce9d   David S. Miller   [SPARC]: Convert ...
589
590
591
592
593
594
  static struct of_platform_driver cg14_driver = {
  	.name		= "cg14",
  	.match_table	= cg14_match,
  	.probe		= cg14_probe,
  	.remove		= __devexit_p(cg14_remove),
  };
1da177e4c   Linus Torvalds   Linux-2.6.12-rc2
595

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