cg6.c

来自「Linux Kernel 2.6.9 for OMAP1710」· C语言 代码 · 共 804 行 · 第 1/2 页

C
804
字号
	spin_unlock_irqrestore(&par->lock, flags);}/** *      cg6_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 cg6_setcolreg(unsigned regno,			 unsigned red, unsigned green, unsigned blue,			 unsigned transp, struct fb_info *info){	struct cg6_par *par = (struct cg6_par *) info->par;	struct bt_regs *bt = par->bt;	unsigned long flags;	if (regno >= 256)		return 1;	red >>= 8;	green >>= 8;	blue >>= 8;	spin_lock_irqsave(&par->lock, flags);	sbus_writel((u32)regno << 24, &bt->addr);	sbus_writel((u32)red << 24, &bt->color_map);	sbus_writel((u32)green << 24, &bt->color_map);	sbus_writel((u32)blue << 24, &bt->color_map);	spin_unlock_irqrestore(&par->lock, flags);	return 0;}/** *      cg6_blank - Optional function.  Blanks the display. *      @blank_mode: the blank mode we want. *      @info: frame buffer structure that represents a single frame buffer */static intcg6_blank(int blank, struct fb_info *info){	struct cg6_par *par = (struct cg6_par *) info->par;	struct cg6_thc *thc = par->thc;	unsigned long flags;	u32 val;	spin_lock_irqsave(&par->lock, flags);	switch (blank) {	case 0: /* Unblanking */		val = sbus_readl(&thc->thc_misc);		val |= CG6_THC_MISC_VIDEO;		sbus_writel(val, &thc->thc_misc);		par->flags &= ~CG6_FLAG_BLANKED;		break;	case 1: /* Normal blanking */	case 2: /* VESA blank (vsync off) */	case 3: /* VESA blank (hsync off) */	case 4: /* Poweroff */		val = sbus_readl(&thc->thc_misc);		val &= ~CG6_THC_MISC_VIDEO;		sbus_writel(val, &thc->thc_misc);		par->flags |= CG6_FLAG_BLANKED;		break;	}	spin_unlock_irqrestore(&par->lock, flags);	return 0;}static struct sbus_mmap_map cg6_mmap_map[] = {	{		.voff	= CG6_FBC,		.poff	= CG6_FBC_OFFSET,		.size	= PAGE_SIZE	},	{		.voff	= CG6_TEC,		.poff	= CG6_TEC_OFFSET,		.size	= PAGE_SIZE	},	{		.voff	= CG6_BTREGS,		.poff	= CG6_BROOKTREE_OFFSET,		.size	= PAGE_SIZE	},	{		.voff	= CG6_FHC,		.poff	= CG6_FHC_OFFSET,		.size	= PAGE_SIZE	},	{		.voff	= CG6_THC,		.poff	= CG6_THC_OFFSET,		.size	= PAGE_SIZE	},	{		.voff	= CG6_ROM,		.poff	= CG6_ROM_OFFSET,		.size	= 0x10000	},	{		.voff	= CG6_RAM,		.poff	= CG6_RAM_OFFSET,		.size	= SBUS_MMAP_FBSIZE(1)	},	{		.voff	= CG6_DHC,		.poff	= CG6_DHC_OFFSET,		.size	= 0x40000	},	{ .size	= 0 }};static int cg6_mmap(struct fb_info *info, struct file *file, struct vm_area_struct *vma){	struct cg6_par *par = (struct cg6_par *)info->par;	return sbusfb_mmap_helper(cg6_mmap_map,				  par->physbase, par->fbsize,				  par->sdev->reg_addrs[0].which_io,				  vma);}static int cg6_ioctl(struct inode *inode, struct file *file, unsigned int cmd,		     unsigned long arg, struct fb_info *info){	struct cg6_par *par = (struct cg6_par *) info->par;	return sbusfb_ioctl_helper(cmd, arg, info,				   FBTYPE_SUNFAST_COLOR, 8, par->fbsize);}/* *  Initialisation */static voidcg6_init_fix(struct fb_info *info, int linebytes){	struct cg6_par *par = (struct cg6_par *)info->par;	const char *cg6_cpu_name, *cg6_card_name;	u32 conf;	conf = sbus_readl(par->fhc);	switch(conf & CG6_FHC_CPU_MASK) {	case CG6_FHC_CPU_SPARC:		cg6_cpu_name = "sparc";		break;	case CG6_FHC_CPU_68020:		cg6_cpu_name = "68020";		break;	default:		cg6_cpu_name = "i386";		break;	};	if (((conf >> CG6_FHC_REV_SHIFT) & CG6_FHC_REV_MASK) >= 11) {		if (par->fbsize <= 0x100000) {			cg6_card_name = "TGX";		} else {			cg6_card_name = "TGX+";		}	} else {		if (par->fbsize <= 0x100000) {			cg6_card_name = "GX";		} else {			cg6_card_name = "GX+";		}	}	sprintf(info->fix.id, "%s %s", cg6_card_name, cg6_cpu_name);	info->fix.id[sizeof(info->fix.id)-1] = 0;	info->fix.type = FB_TYPE_PACKED_PIXELS;	info->fix.visual = FB_VISUAL_PSEUDOCOLOR;	info->fix.line_length = linebytes;	info->fix.accel = FB_ACCEL_SUN_CGSIX;}/* Initialize Brooktree DAC */static void cg6_bt_init(struct cg6_par *par){	struct bt_regs *bt = par->bt;	sbus_writel(0x04 << 24, &bt->addr);         /* color planes */	sbus_writel(0xff << 24, &bt->control);	sbus_writel(0x05 << 24, &bt->addr);	sbus_writel(0x00 << 24, &bt->control);	sbus_writel(0x06 << 24, &bt->addr);         /* overlay plane */	sbus_writel(0x73 << 24, &bt->control);	sbus_writel(0x07 << 24, &bt->addr);	sbus_writel(0x00 << 24, &bt->control);}static void cg6_chip_init(struct fb_info *info){	struct cg6_par *par = (struct cg6_par *) info->par;	struct cg6_tec *tec = par->tec;	struct cg6_fbc *fbc = par->fbc;	u32 rev, conf, mode, tmp;	int i;		/* Turn off stuff in the Transform Engine. */	sbus_writel(0, &tec->tec_matrix);	sbus_writel(0, &tec->tec_clip);	sbus_writel(0, &tec->tec_vdc);	/* Take care of bugs in old revisions. */	rev = (sbus_readl(par->fhc) >> CG6_FHC_REV_SHIFT) & CG6_FHC_REV_MASK;	if (rev < 5) {		conf = (sbus_readl(par->fhc) & CG6_FHC_RES_MASK) |			CG6_FHC_CPU_68020 | CG6_FHC_TEST |			(11 << CG6_FHC_TEST_X_SHIFT) |			(11 << CG6_FHC_TEST_Y_SHIFT);		if (rev < 2)			conf |= CG6_FHC_DST_DISABLE;		sbus_writel(conf, par->fhc);	}	/* Set things in the FBC. Bad things appear to happen if we do	 * back to back store/loads on the mode register, so copy it	 * out instead. */	mode = sbus_readl(&fbc->mode);	do {		i = sbus_readl(&fbc->s);	} while (i & 0x10000000);	mode &= ~(CG6_FBC_BLIT_MASK | CG6_FBC_MODE_MASK |		       CG6_FBC_DRAW_MASK | CG6_FBC_BWRITE0_MASK |		       CG6_FBC_BWRITE1_MASK | CG6_FBC_BREAD_MASK |		       CG6_FBC_BDISP_MASK);	mode |= (CG6_FBC_BLIT_SRC | CG6_FBC_MODE_COLOR8 |		      CG6_FBC_DRAW_RENDER | CG6_FBC_BWRITE0_ENABLE |		      CG6_FBC_BWRITE1_DISABLE | CG6_FBC_BREAD_0 |		      CG6_FBC_BDISP_0);	sbus_writel(mode, &fbc->mode);	sbus_writel(0, &fbc->clip);	sbus_writel(0, &fbc->offx);	sbus_writel(0, &fbc->offy);	sbus_writel(0, &fbc->clipminx);	sbus_writel(0, &fbc->clipminy);	sbus_writel(info->var.xres - 1, &fbc->clipmaxx);	sbus_writel(info->var.yres - 1, &fbc->clipmaxy);	/* Disable cursor in Brooktree DAC. */	sbus_writel(0x06 << 24, &par->bt->addr);	tmp = sbus_readl(&par->bt->control);	tmp &= ~(0x03 << 24);	sbus_writel(tmp, &par->bt->control);}struct all_info {	struct fb_info info;	struct cg6_par par;	struct list_head list;};static LIST_HEAD(cg6_list);static void cg6_init_one(struct sbus_dev *sdev){	struct all_info *all;	int linebytes;	all = kmalloc(sizeof(*all), GFP_KERNEL);	if (!all) {		printk(KERN_ERR "cg6: Cannot allocate memory.\n");		return;	}	memset(all, 0, sizeof(*all));	INIT_LIST_HEAD(&all->list);	spin_lock_init(&all->par.lock);	all->par.sdev = sdev;	all->par.physbase = sdev->reg_addrs[0].phys_addr;	sbusfb_fill_var(&all->info.var, sdev->prom_node, 8);	linebytes = prom_getintdefault(sdev->prom_node, "linebytes",				       all->info.var.xres);	all->par.fbsize = PAGE_ALIGN(linebytes * all->info.var.yres);	if (prom_getbool(sdev->prom_node, "dblbuf"))		all->par.fbsize *= 4;	all->par.fbc = (struct cg6_fbc *)		sbus_ioremap(&sdev->resource[0], CG6_FBC_OFFSET,			     4096, "cgsix fbc");	all->par.tec = (struct cg6_tec *)		sbus_ioremap(&sdev->resource[0], CG6_TEC_OFFSET,			     sizeof(struct cg6_tec), "cgsix tec");	all->par.thc = (struct cg6_thc *)		sbus_ioremap(&sdev->resource[0], CG6_THC_OFFSET,			     sizeof(struct cg6_thc), "cgsix thc");	all->par.bt = (struct bt_regs *)		sbus_ioremap(&sdev->resource[0], CG6_BROOKTREE_OFFSET,			     sizeof(struct bt_regs), "cgsix dac");	all->par.fhc = (u32 *)		sbus_ioremap(&sdev->resource[0], CG6_FHC_OFFSET,			     sizeof(u32), "cgsix fhc");	all->info.flags = FBINFO_DEFAULT | FBINFO_HWACCEL_IMAGEBLIT |                          FBINFO_HWACCEL_COPYAREA | FBINFO_HWACCEL_FILLRECT;	all->info.fbops = &cg6_ops;#ifdef CONFIG_SPARC32	all->info.screen_base = (char *)		prom_getintdefault(sdev->prom_node, "address", 0);#endif	if (!all->info.screen_base)		all->info.screen_base = (char *)			sbus_ioremap(&sdev->resource[0], CG6_RAM_OFFSET,				     all->par.fbsize, "cgsix ram");	all->info.currcon = -1;	all->info.par = &all->par;	all->info.var.accel_flags = FB_ACCELF_TEXT;	cg6_bt_init(&all->par);	cg6_chip_init(&all->info);	cg6_blank(0, &all->info);	if (fb_alloc_cmap(&all->info.cmap, 256, 0)) {		printk(KERN_ERR "cg6: Could not allocate color map.\n");		kfree(all);		return;	}	cg6_init_fix(&all->info, linebytes);	if (register_framebuffer(&all->info) < 0) {		printk(KERN_ERR "cg6: Could not register framebuffer.\n");		fb_dealloc_cmap(&all->info.cmap);		kfree(all);		return;	}	list_add(&all->list, &cg6_list);	printk("cg6: CGsix [%s] at %lx:%lx\n",	       all->info.fix.id,	       (long) sdev->reg_addrs[0].which_io,	       (long) sdev->reg_addrs[0].phys_addr);}int __init cg6_init(void){	struct sbus_bus *sbus;	struct sbus_dev *sdev;	if (fb_get_options("cg6fb", NULL))		return -ENODEV;	for_all_sbusdev(sdev, sbus) {		if (!strcmp(sdev->prom_name, "cgsix") ||		    !strcmp(sdev->prom_name, "cgthree+"))			cg6_init_one(sdev);	}	return 0;}void __exit cg6_exit(void){	struct list_head *pos, *tmp;	list_for_each_safe(pos, tmp, &cg6_list) {		struct all_info *all = list_entry(pos, typeof(*all), list);		unregister_framebuffer(&all->info);		fb_dealloc_cmap(&all->info.cmap);		kfree(all);	}}int __initcg6_setup(char *arg){	/* No cmdline options yet... */	return 0;}module_init(cg6_init);#ifdef MODULEmodule_exit(cg6_exit);#endifMODULE_DESCRIPTION("framebuffer driver for CGsix chipsets");MODULE_AUTHOR("David S. Miller <davem@redhat.com>");MODULE_LICENSE("GPL");

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?