platinumfb.c

来自「Linux环境下视频显示卡设备的驱动程序源代码」· C语言 代码 · 共 709 行 · 第 1/2 页

C
709
字号
/* *  platinumfb.c -- frame buffer device for the PowerMac 'platinum' display * *  Copyright (C) 1998 Franz Sirl * *  Frame buffer structure from: *    drivers/video/controlfb.c -- frame buffer device for *    Apple 'control' display chip. *    Copyright (C) 1998 Dan Jacobowitz * *  Hardware information from: *    platinum.c: Console support for PowerMac "platinum" display adaptor. *    Copyright (C) 1996 Paul Mackerras and Mark Abene * *  This file is subject to the terms and conditions of the GNU General Public *  License. See the file COPYING in the main directory of this archive for *  more details. */#undef DEBUG#include <linux/module.h>#include <linux/kernel.h>#include <linux/errno.h>#include <linux/string.h>#include <linux/mm.h>#include <linux/slab.h>#include <linux/vmalloc.h>#include <linux/delay.h>#include <linux/interrupt.h>#include <linux/fb.h>#include <linux/init.h>#include <linux/nvram.h>#include <linux/of_device.h>#include <linux/of_platform.h>#include <asm/io.h>#include <asm/prom.h>#include <asm/pgtable.h>#include "macmodes.h"#include "platinumfb.h"static int default_vmode = VMODE_NVRAM;static int default_cmode = CMODE_NVRAM;struct fb_info_platinum {	struct fb_info			*info;	int				vmode, cmode;	int				xres, yres;	int				vxres, vyres;	int				xoffset, yoffset;	struct {		__u8 red, green, blue;	}				palette[256];	u32				pseudo_palette[16];		volatile struct cmap_regs	__iomem *cmap_regs;	unsigned long			cmap_regs_phys;		volatile struct platinum_regs	__iomem *platinum_regs;	unsigned long			platinum_regs_phys;		__u8				__iomem *frame_buffer;	volatile __u8			__iomem *base_frame_buffer;	unsigned long			frame_buffer_phys;		unsigned long			total_vram;	int				clktype;	int				dactype;	struct resource			rsrc_fb, rsrc_reg;};/* * Frame buffer device API */static int platinumfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue,	u_int transp, struct fb_info *info);static int platinumfb_blank(int blank_mode, struct fb_info *info);static int platinumfb_set_par (struct fb_info *info);static int platinumfb_check_var (struct fb_var_screeninfo *var, struct fb_info *info);/* * internal functions */static inline int platinum_vram_reqd(int video_mode, int color_mode);static int read_platinum_sense(struct fb_info_platinum *pinfo);static void set_platinum_clock(struct fb_info_platinum *pinfo);static void platinum_set_hardware(struct fb_info_platinum *pinfo);static int platinum_var_to_par(struct fb_var_screeninfo *var,			       struct fb_info_platinum *pinfo,			       int check_only);/* * Interface used by the world */static struct fb_ops platinumfb_ops = {	.owner =	THIS_MODULE,	.fb_check_var	= platinumfb_check_var,	.fb_set_par	= platinumfb_set_par,	.fb_setcolreg	= platinumfb_setcolreg,	.fb_blank	= platinumfb_blank,	.fb_fillrect	= cfb_fillrect,	.fb_copyarea	= cfb_copyarea,	.fb_imageblit	= cfb_imageblit,};/* * Checks a var structure */static int platinumfb_check_var (struct fb_var_screeninfo *var, struct fb_info *info){	return platinum_var_to_par(var, info->par, 1);}/* * Applies current var to display */static int platinumfb_set_par (struct fb_info *info){	struct fb_info_platinum *pinfo = info->par;	struct platinum_regvals *init;	int err, offset = 0x20;	if((err = platinum_var_to_par(&info->var, pinfo, 0))) {		printk (KERN_ERR "platinumfb_set_par: error calling"				 " platinum_var_to_par: %d.\n", err);		return err;	}	platinum_set_hardware(pinfo);	init = platinum_reg_init[pinfo->vmode-1];	 	if ((pinfo->vmode == VMODE_832_624_75) && (pinfo->cmode > CMODE_8))  		offset = 0x10;	info->screen_base = pinfo->frame_buffer + init->fb_offset + offset;	info->fix.smem_start = (pinfo->frame_buffer_phys) + init->fb_offset + offset;	info->fix.visual = (pinfo->cmode == CMODE_8) ?		FB_VISUAL_PSEUDOCOLOR : FB_VISUAL_DIRECTCOLOR; 	info->fix.line_length = vmode_attrs[pinfo->vmode-1].hres * (1<<pinfo->cmode)		+ offset;	printk("line_length: %x\n", info->fix.line_length);	return 0;}static int platinumfb_blank(int blank,  struct fb_info *fb){/* *  Blank the screen if blank_mode != 0, else unblank. If blank == NULL *  then the caller blanks by setting the CLUT (Color Look Up Table) to all *  black. Return 0 if blanking succeeded, != 0 if un-/blanking failed due *  to e.g. a video mode which doesn't support it. Implements VESA suspend *  and powerdown modes on hardware that supports disabling hsync/vsync: *    blank_mode == 2: suspend vsync *    blank_mode == 3: suspend hsync *    blank_mode == 4: powerdown *//* [danj] I think there's something fishy about those constants... *//*	struct fb_info_platinum *info = (struct fb_info_platinum *) fb;	int	ctrl;	ctrl = ld_le32(&info->platinum_regs->ctrl.r) | 0x33;	if (blank)		--blank_mode;	if (blank & VESA_VSYNC_SUSPEND)		ctrl &= ~3;	if (blank & VESA_HSYNC_SUSPEND)		ctrl &= ~0x30;	out_le32(&info->platinum_regs->ctrl.r, ctrl);*//* TODO: Figure out how the heck to powerdown this thing! */	return 0;}static int platinumfb_setcolreg(u_int regno, u_int red, u_int green, u_int blue,			      u_int transp, struct fb_info *info){	struct fb_info_platinum *pinfo = info->par;	volatile struct cmap_regs __iomem *cmap_regs = pinfo->cmap_regs;	if (regno > 255)		return 1;	red >>= 8;	green >>= 8;	blue >>= 8;	pinfo->palette[regno].red = red;	pinfo->palette[regno].green = green;	pinfo->palette[regno].blue = blue;	out_8(&cmap_regs->addr, regno);		/* tell clut what addr to fill	*/	out_8(&cmap_regs->lut, red);		/* send one color channel at	*/	out_8(&cmap_regs->lut, green);		/* a time...			*/	out_8(&cmap_regs->lut, blue);	if (regno < 16) {		int i;		u32 *pal = info->pseudo_palette;		switch (pinfo->cmode) {		case CMODE_16:			pal[regno] = (regno << 10) | (regno << 5) | regno;			break;		case CMODE_32:			i = (regno << 8) | regno;			pal[regno] = (i << 16) | i;			break;		}	}		return 0;}static inline int platinum_vram_reqd(int video_mode, int color_mode){	return vmode_attrs[video_mode-1].vres *	       (vmode_attrs[video_mode-1].hres * (1<<color_mode) +		((video_mode == VMODE_832_624_75) &&		 (color_mode > CMODE_8)) ? 0x10 : 0x20) + 0x1000;}#define STORE_D2(a, d) { \	out_8(&cmap_regs->addr, (a+32)); \	out_8(&cmap_regs->d2, (d)); \}static void set_platinum_clock(struct fb_info_platinum *pinfo){	volatile struct cmap_regs __iomem *cmap_regs = pinfo->cmap_regs;	struct platinum_regvals	*init;	init = platinum_reg_init[pinfo->vmode-1];	STORE_D2(6, 0xc6);	out_8(&cmap_regs->addr,3+32);	if (in_8(&cmap_regs->d2) == 2) {		STORE_D2(7, init->clock_params[pinfo->clktype][0]);		STORE_D2(8, init->clock_params[pinfo->clktype][1]);		STORE_D2(3, 3);	} else {		STORE_D2(4, init->clock_params[pinfo->clktype][0]);		STORE_D2(5, init->clock_params[pinfo->clktype][1]);		STORE_D2(3, 2);	}	__delay(5000);	STORE_D2(9, 0xa6);}/* Now how about actually saying, Make it so! *//* Some things in here probably don't need to be done each time. */static void platinum_set_hardware(struct fb_info_platinum *pinfo){	volatile struct platinum_regs	__iomem *platinum_regs = pinfo->platinum_regs;	volatile struct cmap_regs	__iomem *cmap_regs = pinfo->cmap_regs;	struct platinum_regvals		*init;	int				i;	int				vmode, cmode;		vmode = pinfo->vmode;	cmode = pinfo->cmode;	init = platinum_reg_init[vmode - 1];	/* Initialize display timing registers */	out_be32(&platinum_regs->reg[24].r, 7);	/* turn display off */	for (i = 0; i < 26; ++i)		out_be32(&platinum_regs->reg[i+32].r, init->regs[i]);	out_be32(&platinum_regs->reg[26+32].r, (pinfo->total_vram == 0x100000 ?						init->offset[cmode] + 4 - cmode :						init->offset[cmode]));	out_be32(&platinum_regs->reg[16].r, (unsigned) pinfo->frame_buffer_phys+init->fb_offset+0x10);	out_be32(&platinum_regs->reg[18].r, init->pitch[cmode]);	out_be32(&platinum_regs->reg[19].r, (pinfo->total_vram == 0x100000 ?					     init->mode[cmode+1] :					     init->mode[cmode]));	out_be32(&platinum_regs->reg[20].r, (pinfo->total_vram == 0x100000 ? 0x11 : 0x1011));	out_be32(&platinum_regs->reg[21].r, 0x100);	out_be32(&platinum_regs->reg[22].r, 1);	out_be32(&platinum_regs->reg[23].r, 1);	out_be32(&platinum_regs->reg[26].r, 0xc00);	out_be32(&platinum_regs->reg[27].r, 0x235);	/* out_be32(&platinum_regs->reg[27].r, 0x2aa); */	STORE_D2(0, (pinfo->total_vram == 0x100000 ?		     init->dacula_ctrl[cmode] & 0xf :		     init->dacula_ctrl[cmode]));	STORE_D2(1, 4);	STORE_D2(2, 0);	set_platinum_clock(pinfo);	out_be32(&platinum_regs->reg[24].r, 0);	/* turn display on */}/* * Set misc info vars for this driver */static void __devinit platinum_init_info(struct fb_info *info, struct fb_info_platinum *pinfo){	/* Fill fb_info */	info->fbops = &platinumfb_ops;	info->pseudo_palette = pinfo->pseudo_palette;        info->flags = FBINFO_DEFAULT;	info->screen_base = pinfo->frame_buffer + 0x20;	fb_alloc_cmap(&info->cmap, 256, 0);	/* Fill fix common fields */	strcpy(info->fix.id, "platinum");	info->fix.mmio_start = pinfo->platinum_regs_phys;	info->fix.mmio_len = 0x1000;	info->fix.type = FB_TYPE_PACKED_PIXELS;	info->fix.smem_start = pinfo->frame_buffer_phys + 0x20; /* will be updated later */	info->fix.smem_len = pinfo->total_vram - 0x20;        info->fix.ywrapstep = 0;	info->fix.xpanstep = 0;	info->fix.ypanstep = 0;        info->fix.type_aux = 0;        info->fix.accel = FB_ACCEL_NONE;}static int __devinit platinum_init_fb(struct fb_info *info){	struct fb_info_platinum *pinfo = info->par;	struct fb_var_screeninfo var;	int sense, rc;	sense = read_platinum_sense(pinfo);	printk(KERN_INFO "platinumfb: Monitor sense value = 0x%x, ", sense);	if (default_vmode == VMODE_NVRAM) {#ifdef CONFIG_NVRAM		default_vmode = nvram_read_byte(NV_VMODE);		if (default_vmode <= 0 || default_vmode > VMODE_MAX ||		    !platinum_reg_init[default_vmode-1])#endif			default_vmode = VMODE_CHOOSE;	}	if (default_vmode == VMODE_CHOOSE) {		default_vmode = mac_map_monitor_sense(sense);	}	if (default_vmode <= 0 || default_vmode > VMODE_MAX)

⌨️ 快捷键说明

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