cyber2000fb.c
来自「linux下的VIDEO接口驱动程序」· C语言 代码 · 共 1,976 行 · 第 1/4 页
C
1,976 行
/* * linux/drivers/video/cyber2000fb.c * * Copyright (C) 1998-2002 Russell King * * MIPS and 50xx clock support * Copyright (C) 2001 Bradley D. LaRonde <brad@ltc.com> * * 32 bit support, text color and panning fixes for modes != 8 bit * Copyright (C) 2002 Denis Oliver Kropp <dok@directfb.org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * Integraphics CyberPro 2000, 2010 and 5000 frame buffer device * * Based on cyberfb.c. * * Note that we now use the new fbcon fix, var and cmap scheme. We do * still have to check which console is the currently displayed one * however, especially for the colourmap stuff. * * We also use the new hotplug PCI subsystem. I'm not sure if there * are any such cards, but I'm erring on the side of caution. We don't * want to go pop just because someone does have one. * * Note that this doesn't work fully in the case of multiple CyberPro * cards with grabbers. We currently can only attach to the first * CyberPro card found. * * When we're in truecolour mode, we power down the LUT RAM as a power * saving feature. Also, when we enter any of the powersaving modes * (except soft blanking) we power down the RAMDACs. This saves about * 1W, which is roughly 8% of the power consumption of a NetWinder * (which, incidentally, is about the same saving as a 2.5in hard disk * entering standby mode.) */#include <linux/config.h>#include <linux/module.h>#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/fb.h>#include <linux/pci.h>#include <linux/init.h>#include <asm/io.h>#include <asm/irq.h>#include <asm/pgtable.h>#include <asm/system.h>#include <asm/uaccess.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 "cyber2000fb.h"struct cfb_info { struct fb_info fb; struct display_switch *dispsw; struct display *display; struct pci_dev *dev; unsigned char *region; unsigned char *regs; u_int id; int func_use_count; u_long ref_ps; /* * Clock divisors */ u_int divisors[4]; struct { u8 red, green, blue; } palette[NR_PALETTE]; u_char mem_ctl0; u_char mem_ctl1; u_char mem_ctl2; u_char mclk_mult; u_char mclk_div; /* * RAMDAC control register is both of these or'ed together */ u_char ramdac_ctrl; u_char ramdac_powerdown;};static char default_font_storage[40];static char *default_font = "Acorn8x8";MODULE_PARM(default_font, "s");MODULE_PARM_DESC(default_font, "Default font name");/* * Our access methods. */#define cyber2000fb_writel(val,reg,cfb) writel(val, (cfb)->regs + (reg))#define cyber2000fb_writew(val,reg,cfb) writew(val, (cfb)->regs + (reg))#define cyber2000fb_writeb(val,reg,cfb) writeb(val, (cfb)->regs + (reg))#define cyber2000fb_readb(reg,cfb) readb((cfb)->regs + (reg))static inline voidcyber2000_crtcw(unsigned int reg, unsigned int val, struct cfb_info *cfb){ cyber2000fb_writew((reg & 255) | val << 8, 0x3d4, cfb);}static inline voidcyber2000_grphw(unsigned int reg, unsigned int val, struct cfb_info *cfb){ cyber2000fb_writew((reg & 255) | val << 8, 0x3ce, cfb);}static inline unsigned intcyber2000_grphr(unsigned int reg, struct cfb_info *cfb){ cyber2000fb_writeb(reg, 0x3ce, cfb); return cyber2000fb_readb(0x3cf, cfb);}static inline voidcyber2000_attrw(unsigned int reg, unsigned int val, struct cfb_info *cfb){ cyber2000fb_readb(0x3da, cfb); cyber2000fb_writeb(reg, 0x3c0, cfb); cyber2000fb_readb(0x3c1, cfb); cyber2000fb_writeb(val, 0x3c0, cfb);}static inline voidcyber2000_seqw(unsigned int reg, unsigned int val, struct cfb_info *cfb){ cyber2000fb_writew((reg & 255) | val << 8, 0x3c4, cfb);}/* -------------------- Hardware specific routines ------------------------- *//* * Hardware Cyber2000 Acceleration */static void cyber2000_accel_wait(struct cfb_info *cfb){ int count = 100000; while (cyber2000fb_readb(CO_REG_CONTROL, cfb) & CO_CTRL_BUSY) { if (!count--) { debug_printf("accel_wait timed out\n"); cyber2000fb_writeb(0, CO_REG_CONTROL, cfb); return; } udelay(1); }}static void cyber2000_accel_setup(struct display *display){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; cfb->dispsw->setup(display);}static voidcyber2000_accel_bmove(struct display *display, int sy, int sx, int dy, int dx, int height, int width){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; struct fb_var_screeninfo *var = &display->var; u_long src, dst; u_int fh, fw, cmd = CO_CMD_L_PATTERN_FGCOL; fw = fontwidth(display); sx *= fw; dx *= fw; width *= fw; width -= 1; if (sx < dx) { sx += width; dx += width; cmd |= CO_CMD_L_INC_LEFT; } fh = fontheight(display); sy *= fh; dy *= fh; height *= fh; height -= 1; if (sy < dy) { sy += height; dy += height; cmd |= CO_CMD_L_INC_UP; } src = sx + sy * var->xres_virtual; dst = dx + dy * var->xres_virtual; cyber2000_accel_wait(cfb); cyber2000fb_writeb(0x00, CO_REG_CONTROL, cfb); cyber2000fb_writew(width, CO_REG_PIXWIDTH, cfb); cyber2000fb_writew(height, CO_REG_PIXHEIGHT, cfb); if (var->bits_per_pixel == 24) { cyber2000fb_writeb(dst, CO_REG_X_PHASE, cfb); dst *= 3; src *= 3; } cyber2000fb_writel(src, CO_REG_SRC1_PTR, cfb); cyber2000fb_writel(dst, CO_REG_DEST_PTR, cfb); cyber2000fb_writeb(CO_FG_MIX_SRC, CO_REG_FGMIX, cfb); cyber2000fb_writew(cmd, CO_REG_CMD_L, cfb); cyber2000fb_writew(CO_CMD_H_FGSRCMAP|CO_CMD_H_BLITTER, CO_REG_CMD_H, cfb);}static voidcyber2000_accel_clear(struct vc_data *conp, struct display *display, int sy, int sx, int height, int width){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; struct fb_var_screeninfo *var = &display->var; u_long dst; u_int fw, fh; u32 bgx = attr_bgcol_ec(display, conp); fw = fontwidth(display); fh = fontheight(display); dst = sx * fw + sy * var->xres_virtual * fh; width = width * fw - 1; height = height * fh - 1; cyber2000_accel_wait(cfb); cyber2000fb_writeb(0x00, CO_REG_CONTROL, cfb); cyber2000fb_writew(width, CO_REG_PIXWIDTH, cfb); cyber2000fb_writew(height, CO_REG_PIXHEIGHT, cfb); if (var->bits_per_pixel == 24) { cyber2000fb_writeb(dst, CO_REG_X_PHASE, cfb); dst *= 3; } if (var->bits_per_pixel == 16) bgx = ((u16 *)display->dispsw_data)[bgx]; else if (var->bits_per_pixel >= 24) bgx = ((u32 *)display->dispsw_data)[bgx]; cyber2000fb_writel(bgx, CO_REG_FGCOLOUR, cfb); cyber2000fb_writel(dst, CO_REG_DEST_PTR, cfb); cyber2000fb_writeb(CO_FG_MIX_SRC, CO_REG_FGMIX, cfb); cyber2000fb_writew(CO_CMD_L_PATTERN_FGCOL, CO_REG_CMD_L, cfb); cyber2000fb_writew(CO_CMD_H_BLITTER, CO_REG_CMD_H, cfb);}static voidcyber2000_accel_putc(struct vc_data *conp, struct display *display, int c, int yy, int xx){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; cyber2000_accel_wait(cfb); cfb->dispsw->putc(conp, display, c, yy, xx);}static voidcyber2000_accel_putcs(struct vc_data *conp, struct display *display, const unsigned short *s, int count, int yy, int xx){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; cyber2000_accel_wait(cfb); cfb->dispsw->putcs(conp, display, s, count, yy, xx);}static void cyber2000_accel_revc(struct display *display, int xx, int yy){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; cyber2000_accel_wait(cfb); cfb->dispsw->revc(display, xx, yy);}static voidcyber2000_accel_clear_margins(struct vc_data *conp, struct display *display, int bottom_only){ struct cfb_info *cfb = (struct cfb_info *)display->fb_info; cfb->dispsw->clear_margins(conp, display, bottom_only);}static struct display_switch fbcon_cyber_accel = { .setup = cyber2000_accel_setup, .bmove = cyber2000_accel_bmove, .clear = cyber2000_accel_clear, .putc = cyber2000_accel_putc, .putcs = cyber2000_accel_putcs, .revc = cyber2000_accel_revc, .clear_margins = cyber2000_accel_clear_margins, .fontwidthmask = FONTWIDTH(8)|FONTWIDTH(16)};static inline u32 convert_bitfield(u_int val, struct fb_bitfield *bf){ u_int mask = (1 << bf->length) - 1; return (val >> (16 - bf->length) & mask) << bf->offset;}/* * Set a single color register. Return != 0 for invalid regno. */static intcyber2000fb_setcolreg(u_int regno, u_int red, u_int green, u_int blue, u_int transp, struct fb_info *info){ struct cfb_info *cfb = (struct cfb_info *)info; struct fb_var_screeninfo *var = &cfb->display->var; u32 pseudo_val; int ret = 1; switch (cfb->fb.fix.visual) { default: return 1;#ifdef FBCON_HAS_CFB8 /* * Pseudocolour: * 8 8 * pixel --/--+--/--> red lut --> red dac * | 8 * +--/--> green lut --> green dac * | 8 * +--/--> blue lut --> blue dac */ case FB_VISUAL_PSEUDOCOLOR: if (regno >= NR_PALETTE) return 1; red >>= 8; green >>= 8; blue >>= 8; cfb->palette[regno].red = red; cfb->palette[regno].green = green; cfb->palette[regno].blue = blue; cyber2000fb_writeb(regno, 0x3c8, cfb); cyber2000fb_writeb(red, 0x3c9, cfb); cyber2000fb_writeb(green, 0x3c9, cfb); cyber2000fb_writeb(blue, 0x3c9, cfb); return 0;#endif /* * Direct colour: * n rl * pixel --/--+--/--> red lut --> red dac * | gl * +--/--> green lut --> green dac * | bl * +--/--> blue lut --> blue dac * n = bpp, rl = red length, gl = green length, bl = blue length */ case FB_VISUAL_DIRECTCOLOR: red >>= 8; green >>= 8; blue >>= 8; if (var->green.length == 6 && regno < 64) { cfb->palette[regno << 2].green = green; /* * The 6 bits of the green component are applied * to the high 6 bits of the LUT. */ cyber2000fb_writeb(regno << 2, 0x3c8, cfb); cyber2000fb_writeb(cfb->palette[regno >> 1].red, 0x3c9, cfb); cyber2000fb_writeb(green, 0x3c9, cfb); cyber2000fb_writeb(cfb->palette[regno >> 1].blue, 0x3c9, cfb); green = cfb->palette[regno << 3].green; ret = 0; } if (var->green.length >= 5 && regno < 32) { cfb->palette[regno << 3].red = red; cfb->palette[regno << 3].green = green; cfb->palette[regno << 3].blue = blue; /* * The 5 bits of each colour component are * applied to the high 5 bits of the LUT. */ cyber2000fb_writeb(regno << 3, 0x3c8, cfb); cyber2000fb_writeb(red, 0x3c9, cfb); cyber2000fb_writeb(green, 0x3c9, cfb); cyber2000fb_writeb(blue, 0x3c9, cfb); ret = 0; } if (var->green.length == 4 && regno < 16) { cfb->palette[regno << 4].red = red; cfb->palette[regno << 4].green = green; cfb->palette[regno << 4].blue = blue; /* * The 5 bits of each colour component are * applied to the high 5 bits of the LUT. */ cyber2000fb_writeb(regno << 4, 0x3c8, cfb); cyber2000fb_writeb(red, 0x3c9, cfb); cyber2000fb_writeb(green, 0x3c9, cfb); cyber2000fb_writeb(blue, 0x3c9, cfb); ret = 0; } /* * Since this is only used for the first 16 colours, we * don't have to care about overflowing for regno >= 32 */ pseudo_val = regno << var->red.offset | regno << var->green.offset | regno << var->blue.offset; break; /* * True colour: * n rl * pixel --/--+--/--> red dac * | gl * +--/--> green dac * | bl * +--/--> blue dac * n = bpp, rl = red length, gl = green length, bl = blue length */ case FB_VISUAL_TRUECOLOR: pseudo_val = convert_bitfield(transp ^ 0xffff, &var->transp); pseudo_val |= convert_bitfield(red, &var->red); pseudo_val |= convert_bitfield(green, &var->green); pseudo_val |= convert_bitfield(blue, &var->blue); break; } /* * Now set our pseudo palette for the CFB16/24/32 drivers. */ if (regno < 16) { if (var->bits_per_pixel == 16) ((u16 *)cfb->fb.pseudo_palette)[regno] = pseudo_val; else ((u32 *)cfb->fb.pseudo_palette)[regno] = pseudo_val; ret = 0; } return ret;}struct par_info { /* * Hardware */ u_char clock_mult; u_char clock_div; u_char extseqmisc; u_char co_pixfmt; u_char crtc_ofl; u_char crtc[19]; u_int width; u_int pitch; u_int fetch; /* * Other */ u_char ramdac;};static const u_char crtc_idx[] = { 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18};
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?