⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 tridentfb.c

📁 linux和2410结合开发 用他可以生成2410所需的zImage文件
💻 C
📖 第 1 页 / 共 3 页
字号:
/* * Frame buffer driver for Trident Blade and Image series * * Copyright 2001,2002 - Jani Monoses   <jani@astechnix.ro> * * $Id: tridentfb.c,v 1.2 2002/02/13 17:44:14 marcelo Exp $ * * CREDITS:(in order of appearance) * 	skeletonfb.c by Geert Uytterhoeven and other fb code in drivers/video * 	Special thanks ;) to Mattia Crivellini <tia@mclink.it> * 	much inspired by the XFree86 4.1.0 Trident driver sources by Alan Hourihane * 	the FreeVGA project *	Francesco Salvestrini <salvestrini@users.sf.net> XP support,code,suggestions * NOTES: * 	Tested on Compaq Presario 12XL300 with CyberBladei1 * 	Tested on Toshiba 1800-514 with CyberBladeXPAi1 * 	No monitors were harmed during the writing of this driver * TODO: * 	timing value tweaking so it looks good on every monitor in every mode * 	test text acceleration for the Image series *	test DPMS stuff */#include <linux/config.h>#include <linux/module.h>#include <linux/fb.h>#include <linux/init.h>#include <linux/pci.h>#include <video/fbcon.h>#include <video/fbcon-cfb8.h>#include <video/fbcon-cfb16.h>#include <video/fbcon-cfb24.h>#include <video/fbcon-cfb32.h>#include "tridentfb.h"#define VERSION		"0.6.8"struct tridentfb_par {	struct fb_var_screeninfo var;	int bpp;	int hres;	int vres;	int linelength;	int vclk;		//in MHz	int vtotal;	int vdispend;	int vsyncstart;	int vsyncend;	int vblankstart;	int vblankend;	int htotal;	int hdispend;	int hsyncstart;	int hsyncend;	int hblankstart;	int hblankend;};struct tridentfb_info {	struct fb_info_gen gen;	unsigned int fbmem_virt;	//framebuffer virtual memory address	unsigned int fbmem;		//framebuffer physical memory address	unsigned int memsize;		//size of fbmem	unsigned int io;		//io space address	unsigned int io_virt;		//iospace virtual memory address	unsigned int nativex;		//flat panel xres	struct tridentfb_par currentmode;};static struct fb_ops tridentfb_ops;static struct tridentfb_info fb_info;static struct display disp;static struct { unsigned char red,green,blue,transp; } palette[256];static struct fb_var_screeninfo default_var;static char * tridentfb_name = "Trident";static int family;static int pci_id;static int defaultaccel;static int displaytype;static int pseudo_pal[16];/* defaults which are normally overriden by user values *//* video mode */static char * mode = "640x480";static int bpp = 8;static int noaccel;static int accel;static int center;static int stretch;static int fp;static int crt;static int memsize;static int memdiff;static int nativex;MODULE_PARM(mode,"s");MODULE_PARM(bpp,"i");MODULE_PARM(center,"i");MODULE_PARM(stretch,"i");MODULE_PARM(noaccel,"i");MODULE_PARM(accel,"i");MODULE_PARM(memsize,"i");MODULE_PARM(memdiff,"i");MODULE_PARM(nativex,"i");MODULE_PARM(fp,"i");MODULE_PARM(crt,"i");#define CRT 0x3D0		//CRTC registers offset for color display#ifndef TRIDENT_MMIO	#define TRIDENT_MMIO 1#endif#if TRIDENT_MMIO	#define t_outb(val,reg)	writeb(val,fb_info.io_virt + reg)	#define t_inb(reg)	readb(fb_info.io_virt + reg)#else	#define t_outb(val,reg) outb(val,reg)	#define t_inb(reg) inb(reg)#endifstatic struct accel_switch {	void (*init_accel)(int,int);	void (*wait_engine)(void);	void (*fill_rect)(int,int,int,int,int);	void (*copy_rect)(int,int,int,int,int,int);} *acc;#define writemmr(r,v)	writel(v, fb_info.io_virt + r)#define readmmr(r)	readl(fb_info.io_virt + r)/* * Blade specific acceleration.Not XP's though those are * unaccelerated. */#define point(x,y) ((y)<<16|(x))#define STA	0x2120#define CMD	0x2144#define ROP	0x2148#define CLR	0x2160#define SR1	0x2100#define SR2	0x2104#define DR1	0x2108#define DR2	0x210C#define REPL(x)	x = x | x<<16#define ROP_S	0xCCstatic void blade_init_accel(int pitch,int bpp){	int v1 = (pitch>>3)<<20;	int tmp = 0,v2;	switch (bpp) {		case 8:tmp = 0;break;		case 15:tmp = 5;break;		case 16:tmp = 1;break;		case 24:		case 32:tmp = 2;break;	}	v2 = v1 | (tmp<<29);	writemmr(0x21C0,v2);	writemmr(0x21C4,v2);	writemmr(0x21B8,v2);	writemmr(0x21BC,v2);	writemmr(0x21D0,v1);	writemmr(0x21D4,v1);	writemmr(0x21C8,v1);	writemmr(0x21CC,v1);	writemmr(0x216C,0);}static void blade_wait_engine(void){	while(readmmr(STA) & 0xFA800000);}static void blade_fill_rect(int x,int y,int w,int h,int c){	writemmr(CLR,c);	writemmr(ROP,ROP_S);	writemmr(CMD,0x20000000|1<<19|1<<4|2<<2);	writemmr(DR1,point(x,y));	writemmr(DR2,point(x+w-1,y+h-1));}static void blade_copy_rect(int x1,int y1,int x2,int y2,int w,int h){	int s1,s2,d1,d2;	int direction = 2;	s1 = point(x1,y1);	s2 = point(x1+w-1,y1+h-1);	d1 = point(x2,y2);	d2 = point(x2+w-1,y2+h-1);	if ((y1 > y2) || ((y1 == y2) && (x1 >x2)))			direction = 0;	writemmr(ROP,ROP_S);	writemmr(CMD,0xE0000000|1<<19|1<<4|1<<2|direction);	writemmr(SR1,direction?s2:s1);	writemmr(SR2,direction?s1:s2);	writemmr(DR1,direction?d2:d1);	writemmr(DR2,direction?d1:d2);}static struct accel_switch accel_blade = {	blade_init_accel,	blade_wait_engine,	blade_fill_rect,	blade_copy_rect,};/* * Image specific acceleration functions */static void image_init_accel(int pitch,int bpp){	int tmp = 0;   	switch (bpp) {		case 8:tmp = 0;break;		case 15:tmp = 5;break;		case 16:tmp = 1;break;		case 24:		case 32:tmp = 2;break;	}	writemmr(0x2120, 0xF0000000);	writemmr(0x2120, 0x40000000|tmp);	writemmr(0x2120, 0x80000000);	writemmr(0x2144, 0x00000000);	writemmr(0x2148, 0x00000000);	writemmr(0x2150, 0x00000000);	writemmr(0x2154, 0x00000000);	writemmr(0x2120, 0x60000000 |(pitch<<16) |pitch);	writemmr(0x216C, 0x00000000);	writemmr(0x2170, 0x00000000);	writemmr(0x217C, 0x00000000);	writemmr(0x2120, 0x10000000);	writemmr(0x2130, (2047 << 16) | 2047);}static void image_wait_engine(void){	while(readmmr(0x2164) & 0xF0000000);}static void image_fill_rect(int x,int y,int w,int h,int c){	writemmr(0x2120,0x80000000);	writemmr(0x2120,0x90000000|ROP_S);	writemmr(0x2144,c);	writemmr(DR1,point(x,y));	writemmr(DR2,point(x+w-1,y+h-1));	writemmr(0x2124,0x80000000|3<<22|1<<10|1<<9);}static void image_copy_rect(int x1,int y1,int x2,int y2,int w,int h){	int s1,s2,d1,d2;	int direction = 2;	s1 = point(x1,y1);	s2 = point(x1+w-1,y1+h-1);	d1 = point(x2,y2);	d2 = point(x2+w-1,y2+h-1);	if ((y1 > y2) || ((y1 == y2) && (x1 >x2)))			direction = 0;	writemmr(0x2120,0x80000000);	writemmr(0x2120,0x90000000|ROP_S);	writemmr(SR1,direction?s2:s1);	writemmr(SR2,direction?s1:s2);	writemmr(DR1,direction?d2:d1);	writemmr(DR2,direction?d1:d2);	writemmr(0x2124,0x80000000|1<<22|1<<10|1<<7|direction);}static struct accel_switch accel_image = {	image_init_accel,	image_wait_engine,	image_fill_rect,	image_copy_rect,};/* * Accel functions called by the upper layers */static void trident_bmove (struct display *p, int sy, int sx,				int dy, int dx, int height, int width){	sx *= fontwidth(p);	dx *= fontwidth(p);	width *= fontwidth(p);	sy *= fontheight(p);	dy *= fontheight(p);	height *= fontheight(p);	acc->copy_rect(sx,sy,dx,dy,width,height);	acc->wait_engine();}static void trident_clear_helper (int c, struct display *p,				int sy, int sx, int height, int width){	sx *= fontwidth(p);	sy *= fontheight(p);	width *= fontwidth(p);	height *= fontheight(p);	acc->fill_rect(sx,sy,width,height,c);	acc->wait_engine();}#ifdef FBCON_HAS_CFB8static void trident_8bpp_clear (struct vc_data *conp, struct display *p,				int sy, int sx, int height, int width){	int c;	c = attr_bgcol_ec(p,conp) & 0xFF;	c |= c<<8;	c |= c<<16;	trident_clear_helper(c,p,sy,sx,height,width);}static struct display_switch trident_8bpp = {	setup:		fbcon_cfb8_setup,	bmove:		trident_bmove,	clear:		trident_8bpp_clear,	putc:		fbcon_cfb8_putc,	putcs:		fbcon_cfb8_putcs,	revc:		fbcon_cfb8_revc,	clear_margins:	fbcon_cfb8_clear_margins,	fontwidthmask:	FONTWIDTH (4) | FONTWIDTH (8) | FONTWIDTH (12) | FONTWIDTH (16)};#endif#ifdef FBCON_HAS_CFB16static void trident_16bpp_clear (struct vc_data *conp, struct display *p,				int sy, int sx, int height, int width){	int c;	c = ((u16*)p->dispsw_data)[attr_bgcol_ec(p,conp)];	c = c | c<<16;	trident_clear_helper(c,p,sy,sx,height,width);}static struct display_switch trident_16bpp = {	setup:		fbcon_cfb16_setup,	bmove:		trident_bmove,	clear:		trident_16bpp_clear,	putc:		fbcon_cfb16_putc,	putcs:		fbcon_cfb16_putcs,	revc:		fbcon_cfb16_revc,	clear_margins:	fbcon_cfb16_clear_margins,	fontwidthmask:	FONTWIDTH (4) | FONTWIDTH (8) | FONTWIDTH (12) | FONTWIDTH (16)};#endif#ifdef FBCON_HAS_CFB32static void trident_32bpp_clear (struct vc_data *conp, struct display *p,				int sy, int sx, int height, int width){	int c;	c = ((u32*)p->dispsw_data)[attr_bgcol_ec(p,conp)];	trident_clear_helper(c,p,sy,sx,height,width);}static struct display_switch trident_32bpp = {	setup:		fbcon_cfb32_setup,	bmove:		trident_bmove,	clear:		trident_32bpp_clear,	putc:		fbcon_cfb32_putc,	putcs:		fbcon_cfb32_putcs,	revc:		fbcon_cfb32_revc,	clear_margins:	fbcon_cfb32_clear_margins,	fontwidthmask:	FONTWIDTH (4) | FONTWIDTH (8) | FONTWIDTH (12) | FONTWIDTH (16)};#endif/* * Hardware access functions */static inline unsigned char read3X4(int reg){	writeb(reg, fb_info.io_virt + CRT + 4);	return readb(fb_info.io_virt + CRT + 5);}static inline void write3X4(int reg, unsigned char val){	writeb(reg, fb_info.io_virt + CRT + 4);	writeb(val, fb_info.io_virt + CRT + 5);}static inline unsigned char read3C4(int reg){	t_outb(reg, 0x3C4);	return t_inb(0x3C5);}static inline void write3C4(int reg, unsigned char val){	t_outb(reg, 0x3C4);	t_outb(val, 0x3C5);}static inline unsigned char read3CE(int reg){	t_outb(reg, 0x3CE);

⌨️ 快捷键说明

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