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

📄 dnfb.c

📁 S3C44B0X下的LCD (framebuffer)驱动资料与相关代码
💻 C
字号:
#include <linux/kernel.h>#include <linux/errno.h>#include <linux/string.h>#include <linux/mm.h>#include <linux/tty.h>#include <linux/slab.h>#include <linux/delay.h>#include <linux/interrupt.h>#include <asm/setup.h>#include <asm/segment.h>#include <asm/system.h>#include <asm/irq.h>#include <asm/amigahw.h>#include <asm/amigaints.h>#include <asm/apollohw.h>#include <linux/fb.h>#include <linux/module.h>#include "dn_accel.h"#include <video/fbcon.h>#include <video/fbcon-mfb.h>/* apollo video HW definitions *//* * Control Registers.   IOBASE + $x * * Note: these are the Memory/IO BASE definitions for a mono card set to the * alternate address * * Control 3A and 3B serve identical functions except that 3A * deals with control 1 and 3b deals with Color LUT reg. */#define AP_IOBASE       0x3b0          /* Base address of 1 plane board. */#define AP_STATUS       isaIO2mem(AP_IOBASE+0) /* Status register.  Read */#define AP_WRITE_ENABLE isaIO2mem(AP_IOBASE+0) /* Write Enable Register Write */#define AP_DEVICE_ID    isaIO2mem(AP_IOBASE+1) /* Device ID Register. Read */#define AP_ROP_1        isaIO2mem(AP_IOBASE+2) /* Raster Operation reg. Write Word */#define AP_DIAG_MEM_REQ isaIO2mem(AP_IOBASE+4) /* Diagnostic Memory Request. Write Word */#define AP_CONTROL_0    isaIO2mem(AP_IOBASE+8) /* Control Register 0.  Read/Write */#define AP_CONTROL_1    isaIO2mem(AP_IOBASE+0xa) /* Control Register 1.  Read/Write */#define AP_CONTROL_3A   isaIO2mem(AP_IOBASE+0xe) /* Control Register 3a. Read/Write */#define AP_CONTROL_2    isaIO2mem(AP_IOBASE+0xc) /* Control Register 2. Read/Write */#define FRAME_BUFFER_START 0x0FA0000#define FRAME_BUFFER_LEN 0x40000/* CREG 0 */#define VECTOR_MODE 0x40 /* 010x.xxxx */#define DBLT_MODE   0x80 /* 100x.xxxx */#define NORMAL_MODE 0xE0 /* 111x.xxxx */#define SHIFT_BITS  0x1F /* xxx1.1111 */        /* other bits are Shift value *//* CREG 1 */#define AD_BLT      0x80 /* 1xxx.xxxx */#define NORMAL      0x80 /* 1xxx.xxxx */   /* What is happening here ?? */#define INVERSE     0x00 /* 0xxx.xxxx */   /* Clearing this reverses the screen */#define PIX_BLT     0x00 /* 0xxx.xxxx */#define AD_HIBIT        0x40 /* xIxx.xxxx */#define ROP_EN          0x10 /* xxx1.xxxx */#define DST_EQ_SRC      0x00 /* xxx0.xxxx */#define nRESET_SYNC     0x08 /* xxxx.1xxx */#define SYNC_ENAB       0x02 /* xxxx.xx1x */#define BLANK_DISP      0x00 /* xxxx.xxx0 */#define ENAB_DISP       0x01 /* xxxx.xxx1 */#define NORM_CREG1      (nRESET_SYNC | SYNC_ENAB | ENAB_DISP) /* no reset sync *//* CREG 2 *//* * Following 3 defines are common to 1, 4 and 8 plane. */#define S_DATA_1s   0x00 /* 00xx.xxxx */ /* set source to all 1's -- vector drawing */#define S_DATA_PIX  0x40 /* 01xx.xxxx */ /* takes source from ls-bits and replicates over 16 bits */#define S_DATA_PLN  0xC0 /* 11xx.xxxx */ /* normal, each data access =16-bits in one plane of image mem *//* CREG 3A/CREG 3B */#       define RESET_CREG 0x80 /* 1000.0000 *//* ROP REG  -  all one nibble *//*      ********* NOTE : this is used r0,r1,r2,r3 *********** */#define ROP(r2,r3,r0,r1) ( (U_SHORT)((r0)|((r1)<<4)|((r2)<<8)|((r3)<<12)) )#define DEST_ZERO               0x0#define SRC_AND_DEST    0x1#define SRC_AND_nDEST   0x2#define SRC                             0x3#define nSRC_AND_DEST   0x4#define DEST                    0x5#define SRC_XOR_DEST    0x6#define SRC_OR_DEST             0x7#define SRC_NOR_DEST    0x8#define SRC_XNOR_DEST   0x9#define nDEST                   0xA#define SRC_OR_nDEST    0xB#define nSRC                    0xC#define nSRC_OR_DEST    0xD#define SRC_NAND_DEST   0xE#define DEST_ONE                0xF#define SWAP(A) ((A>>8) | ((A&0xff) <<8))/* frame buffer operations */static int dn_fb_get_fix(struct fb_fix_screeninfo *fix, int con, 			 struct fb_info *info);static int dn_fb_get_var(struct fb_var_screeninfo *var, int con,			 struct fb_info *info);static int dn_fb_set_var(struct fb_var_screeninfo *var, int isactive,			 struct fb_info *info);static int dn_fb_get_cmap(struct fb_cmap *cmap,int kspc,int con,			  struct fb_info *info);static int dn_fb_set_cmap(struct fb_cmap *cmap,int kspc,int con,			  struct fb_info *info);static int dnfbcon_switch(int con,struct fb_info *info);static int dnfbcon_updatevar(int con,struct fb_info *info);static void dnfbcon_blank(int blank,struct fb_info *info);static void dn_fb_set_disp(int con,struct fb_info *info);static struct display disp[MAX_NR_CONSOLES];static struct fb_info fb_info;static struct fb_ops dn_fb_ops = {	owner:		THIS_MODULE,	fb_get_fix:	dn_fb_get_fix,	fb_get_var:	dn_fb_get_var,	fb_set_var:	dn_fb_set_var,	fb_get_cmap:	dn_fb_get_cmap,	fb_set_cmap:	dn_fb_set_cmap,};static int currcon=0;#define NUM_TOTAL_MODES 1struct fb_var_screeninfo dn_fb_predefined[] = {	{ 0, },};static char dn_fb_name[]="Apollo ";/* accel stuff */#define USE_DN_ACCELstatic struct display_switch dispsw_apollofb;static int dn_fb_get_fix(struct fb_fix_screeninfo *fix, int con,			 struct fb_info *info) {	strcpy(fix->id,"Apollo Mono");	fix->smem_start=(FRAME_BUFFER_START+IO_BASE);	fix->smem_len=FRAME_BUFFER_LEN;	fix->type=FB_TYPE_PACKED_PIXELS;	fix->type_aux=0;	fix->visual=FB_VISUAL_MONO10;	fix->xpanstep=0;	fix->ypanstep=0;	fix->ywrapstep=0;        fix->line_length=256;	return 0;}        static int dn_fb_get_var(struct fb_var_screeninfo *var, int con,			 struct fb_info *info) {			var->xres=1280;	var->yres=1024;	var->xres_virtual=2048;	var->yres_virtual=1024;	var->xoffset=0;	var->yoffset=0;	var->bits_per_pixel=1;	var->grayscale=0;	var->nonstd=0;	var->activate=0;	var->height=-1;	var->width=-1;	var->pixclock=0;	var->left_margin=0;	var->right_margin=0;	var->hsync_len=0;	var->vsync_len=0;	var->sync=0;	var->vmode=FB_VMODE_NONINTERLACED;	return 0;}static int dn_fb_set_var(struct fb_var_screeninfo *var, int con,			 struct fb_info *info) {        printk("fb_set_var\n");	if(var->xres!=1280) 		return -EINVAL;	if(var->yres!=1024)		return -EINVAL;	if(var->xres_virtual!=2048)		return -EINVAL;	if(var->yres_virtual!=1024)		return -EINVAL;	if(var->xoffset!=0)		return -EINVAL;	if(var->yoffset!=0)		return -EINVAL;	if(var->bits_per_pixel!=1)		return -EINVAL;	if(var->grayscale!=0)		return -EINVAL;	if(var->nonstd!=0)		return -EINVAL;	if(var->activate!=0)		return -EINVAL;	if(var->pixclock!=0)		return -EINVAL;	if(var->left_margin!=0)		return -EINVAL;	if(var->right_margin!=0)		return -EINVAL;	if(var->hsync_len!=0)		return -EINVAL;	if(var->vsync_len!=0)		return -EINVAL;	if(var->sync!=0)		return -EINVAL;	if(var->vmode!=FB_VMODE_NONINTERLACED)		return -EINVAL;	return 0;}static int dn_fb_get_cmap(struct fb_cmap *cmap,int kspc,int con,			  struct fb_info *info) {	printk("get cmap not supported\n");	return -EINVAL;}static int dn_fb_set_cmap(struct fb_cmap *cmap,int kspc,int con,			  struct fb_info *info) {	printk("set cmap not supported\n");	return -EINVAL;}static void dn_fb_set_disp(int con, struct fb_info *info) {  struct fb_fix_screeninfo fix;  struct display *display;    dn_fb_get_fix(&fix,con, info);  if (con>=0)	display=&fb_display[con];  else	display=&disp[0];  if(con==-1)     con=0;   display->screen_base = (u_char *)fix.smem_start;   display->visual = fix.visual;   display->type = fix.type;   display->type_aux = fix.type_aux;   display->ypanstep = fix.ypanstep;   display->ywrapstep = fix.ywrapstep;   display->can_soft_blank = 1;   display->inverse = 0;   display->line_length = fix.line_length;#ifdef FBCON_HAS_MFB   display->dispsw = &fbcon_mfb;#else   display->dispsw=&fbcon_dummy;#endif}  unsigned long __init dnfb_init(unsigned long mem_start) {	int err;       	fb_info.changevar=NULL;	strcpy(&fb_info.modename[0],dn_fb_name);	fb_info.fontname[0]=0;	fb_info.disp=disp;	fb_info.switch_con=&dnfbcon_switch;	fb_info.updatevar=&dnfbcon_updatevar;	fb_info.blank=&dnfbcon_blank;		fb_info.node = -1;	fb_info.fbops = &dn_fb_ops;	        dn_fb_get_var(&disp[0].var,0, &fb_info);	dn_fb_set_disp(-1, &fb_info);		err=register_framebuffer(&fb_info);	if(err < 0) {		panic("unable to register apollo frame buffer\n");	} 	/* now we have registered we can safely setup the hardware */        out_8(AP_CONTROL_3A, RESET_CREG);        out_be16(AP_WRITE_ENABLE, 0x0);        out_8(AP_CONTROL_0, NORMAL_MODE);         out_8(AP_CONTROL_1, (AD_BLT | DST_EQ_SRC | NORM_CREG1));        out_8(AP_CONTROL_2, S_DATA_PLN);        out_be16(AP_ROP_1, SWAP(0x3));        printk("apollo frame buffer alive and kicking !\n");	return mem_start;}		static int dnfbcon_switch(int con,  struct fb_info *info) { 	currcon=con;		return 0;}static int dnfbcon_updatevar(int con,  struct fb_info *info) {	return 0;}static void dnfbcon_blank(int blank,  struct fb_info *info) {	if(blank)  {		out_8(AP_CONTROL_3A, 0x0);	}	else {	        out_8(AP_CONTROL_3A, 0x1);	}	return ;}void dn_bitblt(struct display *p,int x_src,int y_src, int x_dest, int y_dest,               int x_count, int y_count) {	int incr,y_delta,pre_read=0,x_end,x_word_count;	ushort *src,dummy;	uint start_mask,end_mask,dest;	short i,j;	incr=(y_dest<=y_src) ? 1 : -1 ;	src=(ushort *)(p->screen_base+ y_src*p->next_line+(x_src >> 4));	dest=y_dest*(p->next_line >> 1)+(x_dest >> 4);		if(incr>0) {		y_delta=(p->next_line*8)-x_src-x_count;		x_end=x_dest+x_count-1;		x_word_count=(x_end>>4) - (x_dest >> 4) + 1;		start_mask=0xffff0000 >> (x_dest & 0xf);		end_mask=0x7ffff >> (x_end & 0xf);		out_8(AP_CONTROL_0,		      (((x_dest & 0xf) - (x_src &0xf))  % 16)|(0x4 << 5));		if((x_dest & 0xf) < (x_src & 0xf))			pre_read=1;	}	else {		y_delta=-((p->next_line*8)-x_src-x_count);		x_end=x_dest-x_count+1;		x_word_count=(x_dest>>4) - (x_end >> 4) + 1;		start_mask=0x7ffff >> (x_dest & 0xf);		end_mask=0xffff0000 >> (x_end & 0xf);		out_8(AP_CONTROL_0,		      ((-((x_src & 0xf) - (x_dest &0xf))) % 16)|(0x4 << 5));		if((x_dest & 0xf) > (x_src & 0xf))			pre_read=1;	}	for(i=0;i<y_count;i++) {		out_8(AP_CONTROL_3A, 0xc | (dest >> 16));		if(pre_read) {			dummy=*src;			src+=incr;		}		if(x_word_count) {			out_8(AP_WRITE_ENABLE, start_mask);			*src=dest;			src+=incr;			dest+=incr;			out_8(AP_WRITE_ENABLE, 0);			for(j=1;j<(x_word_count-1);j++) {				*src=dest;				src+=incr;					dest+=incr;			}			out_8(AP_WRITE_ENABLE, start_mask);			*src=dest;			dest+=incr;			src+=incr;		}		else {			out_8(AP_WRITE_ENABLE, start_mask | end_mask);			*src=dest;			dest+=incr;			src+=incr;		}		src+=(y_delta/16);		dest+=(y_delta/16);	}	out_8(AP_CONTROL_0, NORMAL_MODE);}static void bmove_apollofb(struct display *p, int sy, int sx, int dy, int dx,		      int height, int width){    int fontheight,fontwidth;    fontheight=fontheight(p);    fontwidth=fontwidth(p);#ifdef USE_DN_ACCEL    dn_bitblt(p,sx,sy*fontheight,dx,dy*fontheight,width*fontwidth,	      height*fontheight);#else    u_char *src, *dest;    u_int rows;    if (sx == 0 && dx == 0 && width == p->next_line) {	src = p->screen_base+sy*fontheight*width;	dest = p->screen_base+dy*fontheight*width;	mymemmove(dest, src, height*fontheight*width);    } else if (dy <= sy) {	src = p->screen_base+sy*fontheight*next_line+sx;	dest = p->screen_base+dy*fontheight*next_line+dx;	for (rows = height*fontheight; rows--;) {	    mymemmove(dest, src, width);	    src += p->next_line;	    dest += p->next_line;	}    } else {	src = p->screen_base+((sy+height)*fontheight-1)*p->next_line+sx;	dest = p->screen_base+((dy+height)*fontheight-1)*p->next_line+dx;	for (rows = height*fontheight; rows--;) {	    mymemmove(dest, src, width);	    src -= p->next_line;	    dest -= p->next_line;	}    }#endif}static void clear_apollofb(struct vc_data *conp, struct display *p, int sy, int sx,		      int height, int width){	fbcon_mfb_clear(conp,p,sy,sx,height,width);}static void putc_apollofb(struct vc_data *conp, struct display *p, int c, int yy,		     int xx){	fbcon_mfb_putc(conp,p,c,yy,xx);}static void putcs_apollofb(struct vc_data *conp, struct display *p, const unsigned short *s,		      int count, int yy, int xx){	fbcon_mfb_putcs(conp,p,s,count,yy,xx);}static void rev_char_apollofb(struct display *p, int xx, int yy){	fbcon_mfb_revc(p,xx,yy);}static struct display_switch dispsw_apollofb = {    setup:		fbcon_mfb_setup,    bmove:		bmove_apollofb,    clear:		clear_apollofb,    putc:		putc_apollofb,    putcs:		putcs_apollofb,    revc:		rev_char_apollofb,    fontwidthmask:	FONTWIDTH(8)};MODULE_LICENSE("GPL");

⌨️ 快捷键说明

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