亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? sisusb_con.c

?? 底層驅動開發
?? C
?? 第 1 頁 / 共 3 頁
字號:
/* * sisusb - usb kernel driver for SiS315(E) based USB2VGA dongles * * VGA text mode console part * * Copyright (C) 2005 by Thomas Winischhofer, Vienna, Austria * * If distributed as part of the Linux kernel, this code is licensed under the * terms of the GPL v2. * * Otherwise, the following license terms apply: * * * Redistribution and use in source and binary forms, with or without * * modification, are permitted provided that the following conditions * * are met: * * 1) Redistributions of source code must retain the above copyright * *    notice, this list of conditions and the following disclaimer. * * 2) Redistributions in binary form must reproduce the above copyright * *    notice, this list of conditions and the following disclaimer in the * *    documentation and/or other materials provided with the distribution. * * 3) The name of the author may not be used to endorse or promote products * *    derived from this software without specific psisusbr written permission. * * * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESSED OR * * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * Author: Thomas Winischhofer <thomas@winischhofer.net> * * Portions based on vgacon.c which are *	Created 28 Sep 1997 by Geert Uytterhoeven *      Rewritten by Martin Mares <mj@ucw.cz>, July 1998 *      based on code Copyright (C) 1991, 1992  Linus Torvalds *			    1995  Jay Estabrook * * A note on using in_atomic() in here: We can't handle console * calls from non-schedulable context due to our USB-dependend * nature. For now, this driver just ignores any calls if it * detects this state. * */#include <linux/config.h>#include <linux/version.h>#include <linux/module.h>#include <linux/kernel.h>#include <linux/signal.h>#include <linux/sched.h>#include <linux/fs.h>#include <linux/tty.h>#include <linux/console.h>#include <linux/string.h>#include <linux/kd.h>#include <linux/init.h>#include <linux/slab.h>#include <linux/vt_kern.h>#include <linux/selection.h>#include <linux/spinlock.h>#include <linux/kref.h>#include <linux/smp_lock.h>#include <linux/ioport.h>#include <linux/interrupt.h>#include <linux/vmalloc.h>#include "sisusb.h"#ifdef INCL_SISUSB_CONextern int sisusb_setreg(struct sisusb_usb_data *, int, u8);extern int sisusb_getreg(struct sisusb_usb_data *, int, u8 *);extern int sisusb_setidxreg(struct sisusb_usb_data *, int, u8, u8);extern int sisusb_getidxreg(struct sisusb_usb_data *, int, u8, u8 *);extern int sisusb_setidxregor(struct sisusb_usb_data *, int, u8, u8);extern int sisusb_setidxregand(struct sisusb_usb_data *, int, u8, u8);extern int sisusb_setidxregandor(struct sisusb_usb_data *, int, u8, u8, u8);extern int sisusb_writeb(struct sisusb_usb_data *sisusb, u32 adr, u8 data);extern int sisusb_readb(struct sisusb_usb_data *sisusb, u32 adr, u8 *data);extern int sisusb_writew(struct sisusb_usb_data *sisusb, u32 adr, u16 data);extern int sisusb_readw(struct sisusb_usb_data *sisusb, u32 adr, u16 *data);extern int sisusb_copy_memory(struct sisusb_usb_data *sisusb, char *src,			u32 dest, int length, size_t *bytes_written);extern void sisusb_delete(struct kref *kref);extern int sisusb_reset_text_mode(struct sisusb_usb_data *sisusb, int init);extern int SiSUSBSetMode(struct SiS_Private *SiS_Pr, unsigned short ModeNo);#define sisusbcon_writew(val, addr)	(*(addr) = (val))#define sisusbcon_readw(addr)		(*(addr))#define sisusbcon_memmovew(d, s, c)	memmove(d, s, c)#define sisusbcon_memcpyw(d, s, c)	memcpy(d, s, c)/* vc_data -> sisusb conversion table */static struct sisusb_usb_data *mysisusbs[MAX_NR_CONSOLES];/* Forward declaration */static const struct consw sisusb_con;extern struct semaphore disconnect_sem;static inline voidsisusbcon_memsetw(u16 *s, u16 c, unsigned int count){	count /= 2;	while (count--)		sisusbcon_writew(c, s++);}static inline voidsisusb_initialize(struct sisusb_usb_data *sisusb){	/* Reset cursor and start address */	if (sisusb_setidxreg(sisusb, SISCR, 0x0c, 0x00))		return;	if (sisusb_setidxreg(sisusb, SISCR, 0x0d, 0x00))		return;	if (sisusb_setidxreg(sisusb, SISCR, 0x0e, 0x00))		return;	sisusb_setidxreg(sisusb, SISCR, 0x0f, 0x00);}static inline voidsisusbcon_set_start_address(struct sisusb_usb_data *sisusb, struct vc_data *c){	sisusb->cur_start_addr = (c->vc_visible_origin - sisusb->scrbuf) / 2;	sisusb_setidxreg(sisusb, SISCR, 0x0c, (sisusb->cur_start_addr >> 8));	sisusb_setidxreg(sisusb, SISCR, 0x0d, (sisusb->cur_start_addr & 0xff));}voidsisusb_set_cursor(struct sisusb_usb_data *sisusb, unsigned int location){	if (sisusb->sisusb_cursor_loc == location)		return;	sisusb->sisusb_cursor_loc = location;	/* Hardware bug: Text cursor appears twice or not at all	 * at some positions. Work around it with the cursor skew	 * bits.	 */	if ((location & 0x0007) == 0x0007) {		sisusb->bad_cursor_pos = 1;		location--;		if (sisusb_setidxregandor(sisusb, SISCR, 0x0b, 0x1f, 0x20))			return;	} else if (sisusb->bad_cursor_pos) {		if (sisusb_setidxregand(sisusb, SISCR, 0x0b, 0x1f))			return;		sisusb->bad_cursor_pos = 0;	}	if (sisusb_setidxreg(sisusb, SISCR, 0x0e, (location >> 8)))		return;	sisusb_setidxreg(sisusb, SISCR, 0x0f, (location & 0xff));}static inline struct sisusb_usb_data *sisusb_get_sisusb(unsigned short console){	return mysisusbs[console];}static inline intsisusb_sisusb_valid(struct sisusb_usb_data *sisusb){	if (!sisusb->present || !sisusb->ready || !sisusb->sisusb_dev)		return 0;	return 1;}static struct sisusb_usb_data *sisusb_get_sisusb_lock_and_check(unsigned short console){	struct sisusb_usb_data *sisusb;	/* We can't handle console calls in non-schedulable	 * context due to our locks and the USB transport.	 * So we simply ignore them. This should only affect	 * some calls to printk.	 */	if (in_atomic())		return NULL;	if (!(sisusb = sisusb_get_sisusb(console)))		return NULL;	down(&sisusb->lock);	if (!sisusb_sisusb_valid(sisusb) ||	    !sisusb->havethisconsole[console]) {		up(&sisusb->lock);		return NULL;	}	return sisusb;}static intsisusb_is_inactive(struct vc_data *c, struct sisusb_usb_data *sisusb){	if (sisusb->is_gfx ||	    sisusb->textmodedestroyed ||	    c->vc_mode != KD_TEXT)		return 1;	return 0;}/* con_startup console interface routine */static const char *sisusbcon_startup(void){	return "SISUSBCON";}/* con_init console interface routine */static voidsisusbcon_init(struct vc_data *c, int init){	struct sisusb_usb_data *sisusb;	int cols, rows;	/* This is called by take_over_console(),	 * ie by us/under our control. It is	 * only called after text mode and fonts	 * are set up/restored.	 */	down(&disconnect_sem);	if (!(sisusb = sisusb_get_sisusb(c->vc_num))) {		up(&disconnect_sem);		return;	}	down(&sisusb->lock);	if (!sisusb_sisusb_valid(sisusb)) {		up(&sisusb->lock);		up(&disconnect_sem);		return;	}	c->vc_can_do_color = 1;	c->vc_complement_mask = 0x7700;	c->vc_hi_font_mask = sisusb->current_font_512 ? 0x0800 : 0;	sisusb->haveconsole = 1;	sisusb->havethisconsole[c->vc_num] = 1;	/* We only support 640x400 */	c->vc_scan_lines = 400;	c->vc_font.height = sisusb->current_font_height;	/* We only support width = 8 */	cols = 80;	rows = c->vc_scan_lines / c->vc_font.height;	/* Increment usage count for our sisusb.	 * Doing so saves us from upping/downing	 * the disconnect semaphore; we can't	 * lose our sisusb until this is undone	 * in con_deinit. For all other console	 * interface functions, it suffices to	 * use sisusb->lock and do a quick check	 * of sisusb for device disconnection.	 */	kref_get(&sisusb->kref);	if (!*c->vc_uni_pagedir_loc)		con_set_default_unimap(c);	up(&sisusb->lock);	up(&disconnect_sem);	if (init) {		c->vc_cols = cols;		c->vc_rows = rows;	} else		vc_resize(c, cols, rows);}/* con_deinit console interface routine */static voidsisusbcon_deinit(struct vc_data *c){	struct sisusb_usb_data *sisusb;	int i;	/* This is called by take_over_console()	 * and others, ie not under our control.	 */	down(&disconnect_sem);	if (!(sisusb = sisusb_get_sisusb(c->vc_num))) {		up(&disconnect_sem);		return;	}	down(&sisusb->lock);	/* Clear ourselves in mysisusbs */	mysisusbs[c->vc_num] = NULL;	sisusb->havethisconsole[c->vc_num] = 0;	/* Free our font buffer if all consoles are gone */	if (sisusb->font_backup) {		for(i = 0; i < MAX_NR_CONSOLES; i++) {			if (sisusb->havethisconsole[c->vc_num])				break;		}		if (i == MAX_NR_CONSOLES) {			vfree(sisusb->font_backup);			sisusb->font_backup = NULL;		}	}	up(&sisusb->lock);	/* decrement the usage count on our sisusb */	kref_put(&sisusb->kref, sisusb_delete);	up(&disconnect_sem);}/* interface routine */static u8sisusbcon_build_attr(struct vc_data *c, u8 color, u8 intensity,			    u8 blink, u8 underline, u8 reverse){	u8 attr = color;	if (underline)		attr = (attr & 0xf0) | c->vc_ulcolor;	else if (intensity == 0)		attr = (attr & 0xf0) | c->vc_halfcolor;	if (reverse)		attr = ((attr) & 0x88) |		       ((((attr) >> 4) |		       ((attr) << 4)) & 0x77);	if (blink)		attr ^= 0x80;	if (intensity == 2)		attr ^= 0x08;	return attr;}/* Interface routine */static voidsisusbcon_invert_region(struct vc_data *vc, u16 *p, int count){	/* Invert a region. This is called with a pointer	 * to the console's internal screen buffer. So we	 * simply do the inversion there and rely on	 * a call to putc(s) to update the real screen.	 */	while (count--) {		u16 a = sisusbcon_readw(p);		a = ((a) & 0x88ff)        |		    (((a) & 0x7000) >> 4) |		    (((a) & 0x0700) << 4);		sisusbcon_writew(a, p++);	}}#define SISUSB_VADDR(x,y) \	((u16 *)c->vc_origin + \	(y) * sisusb->sisusb_num_columns + \	(x))#define SISUSB_HADDR(x,y) \	((u16 *)(sisusb->vrambase + (c->vc_origin - sisusb->scrbuf)) + \	(y) * sisusb->sisusb_num_columns + \	(x))/* Interface routine */static voidsisusbcon_putc(struct vc_data *c, int ch, int y, int x){	struct sisusb_usb_data *sisusb;	ssize_t written;	if (!(sisusb = sisusb_get_sisusb_lock_and_check(c->vc_num)))		return;	/* sisusb->lock is down */	/* Don't need to put the character into buffer ourselves,	 * because the vt does this BEFORE calling us.	 */#if 0	sisusbcon_writew(ch, SISUSB_VADDR(x, y));#endif	if (sisusb_is_inactive(c, sisusb)) {		up(&sisusb->lock);		return;	}	sisusb_copy_memory(sisusb, (char *)SISUSB_VADDR(x, y),				(u32)SISUSB_HADDR(x, y), 2, &written);	up(&sisusb->lock);}/* Interface routine */static voidsisusbcon_putcs(struct vc_data *c, const unsigned short *s,		         int count, int y, int x){	struct sisusb_usb_data *sisusb;	ssize_t written;	u16 *dest;	int i;	if (!(sisusb = sisusb_get_sisusb_lock_and_check(c->vc_num)))		return;	/* sisusb->lock is down */	/* Need to put the characters into the buffer ourselves,	 * because the vt does this AFTER calling us.	 */	dest = SISUSB_VADDR(x, y);	for (i = count; i > 0; i--)		sisusbcon_writew(sisusbcon_readw(s++), dest++);	if (sisusb_is_inactive(c, sisusb)) {		up(&sisusb->lock);		return;	}	sisusb_copy_memory(sisusb, (char *)SISUSB_VADDR(x, y),				(u32)SISUSB_HADDR(x, y), count * 2, &written);	up(&sisusb->lock);}/* Interface routine */static voidsisusbcon_clear(struct vc_data *c, int y, int x, int height, int width){	struct sisusb_usb_data *sisusb;	u16 eattr = c->vc_video_erase_char;	ssize_t written;	int i, length, cols;	u16 *dest;	if (width <= 0 || height <= 0)		return;	if (!(sisusb = sisusb_get_sisusb_lock_and_check(c->vc_num)))		return;	/* sisusb->lock is down */	/* Need to clear buffer ourselves, because the vt does	 * this AFTER calling us.	 */	dest = SISUSB_VADDR(x, y);	cols = sisusb->sisusb_num_columns;	if (width > cols)		width = cols;	if (x == 0 && width >= c->vc_cols) {		sisusbcon_memsetw(dest, eattr, height * cols * 2);	} else {		for (i = height; i > 0; i--, dest += cols)			sisusbcon_memsetw(dest, eattr, width * 2);	}	if (sisusb_is_inactive(c, sisusb)) {		up(&sisusb->lock);		return;	}	length = ((height * cols) - x - (cols - width - x)) * 2;	sisusb_copy_memory(sisusb, (unsigned char *)SISUSB_VADDR(x, y),				(u32)SISUSB_HADDR(x, y), length, &written);	up(&sisusb->lock);}/* Interface routine */static voidsisusbcon_bmove(struct vc_data *c, int sy, int sx,			 int dy, int dx, int height, int width){	struct sisusb_usb_data *sisusb;	ssize_t written;	int cols, length;#if 0	u16 *src, *dest;	int i;#endif	if (width <= 0 || height <= 0)		return;	if (!(sisusb = sisusb_get_sisusb_lock_and_check(c->vc_num)))		return;	/* sisusb->lock is down */	cols = sisusb->sisusb_num_columns;	/* Don't need to move data outselves, because	 * vt does this BEFORE calling us.	 * This is only used by vt's insert/deletechar.	 */#if 0	if (sx == 0 && dx == 0 && width >= c->vc_cols && width <= cols) {		sisusbcon_memmovew(SISUSB_VADDR(0, dy), SISUSB_VADDR(0, sy),					height * width * 2);

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
国产成人免费xxxxxxxx| 一本一道波多野结衣一区二区| 国产精品日产欧美久久久久| 欧美少妇xxx| 成人午夜精品在线| 性做久久久久久久久| 国产精品美女一区二区三区 | 亚洲国产wwwccc36天堂| 精品国免费一区二区三区| 欧美丝袜第三区| 91丨porny丨国产入口| 久久成人综合网| 天堂成人免费av电影一区| 亚洲精品一二三区| 欧美国产禁国产网站cc| 精品免费一区二区三区| 91麻豆精品国产91久久久使用方法| 春色校园综合激情亚洲| 韩国v欧美v亚洲v日本v| 免费日本视频一区| 日韩成人免费电影| 亚洲高清免费观看高清完整版在线观看| 欧美国产综合一区二区| 久久亚洲欧美国产精品乐播| 日韩欧美在线网站| 9191成人精品久久| 欧美日韩大陆在线| 欧美日韩成人在线一区| 欧美午夜片在线看| 在线精品视频免费观看| 日本精品视频一区二区| 色婷婷综合久久久| 色综合婷婷久久| 99久久er热在这里只有精品15| 成人免费观看视频| 成人免费观看男女羞羞视频| 国产成人亚洲综合色影视| 精品一区二区三区的国产在线播放| 日韩专区在线视频| 日韩精品福利网| 婷婷亚洲久悠悠色悠在线播放| 五月综合激情日本mⅴ| 亚洲va中文字幕| 婷婷开心久久网| 日韩av中文字幕一区二区 | 综合久久久久久| 中文字幕一区二区不卡| 国产精品对白交换视频| **网站欧美大片在线观看| 中文字幕制服丝袜一区二区三区| 中文字幕制服丝袜成人av | 亚洲小说欧美激情另类| 亚洲综合精品自拍| 日韩激情在线观看| 看电影不卡的网站| 国产成人午夜精品影院观看视频| 国产不卡视频一区| 91免费观看视频在线| 欧美性videosxxxxx| 日韩一区二区在线观看| 日韩三级在线观看| 日韩午夜电影av| 色狠狠一区二区三区香蕉| 狠狠色综合日日| 成人晚上爱看视频| 国产欧美一区二区三区在线老狼| 亚洲国产色一区| 欧美一区二区三区公司| 国产精品白丝av| 亚洲综合免费观看高清完整版在线| 欧美一区二区在线播放| 国产盗摄一区二区| 亚洲成人黄色小说| 国产精品网友自拍| 91精品国产综合久久婷婷香蕉| 国产乱子伦视频一区二区三区| 亚洲精品国产精品乱码不99| 日韩精品在线看片z| 91久久精品一区二区三| 国产一区美女在线| 香港成人在线视频| 国产精品久久毛片| 精品久久国产老人久久综合| 91啪亚洲精品| 国产精品综合av一区二区国产馆| 亚洲国产婷婷综合在线精品| 国产三级欧美三级日产三级99| 欧美理论片在线| 91网上在线视频| 国产成人高清视频| 日本麻豆一区二区三区视频| 亚洲综合一区二区精品导航| 久久久www成人免费毛片麻豆 | 欧美高清视频在线高清观看mv色露露十八 | 日本一区二区免费在线观看视频| 欧美狂野另类xxxxoooo| 91丨九色porny丨蝌蚪| 国产成人精品亚洲日本在线桃色| 日韩不卡手机在线v区| 一区二区三区四区不卡视频| 国产精品欧美一级免费| 久久先锋资源网| 精品播放一区二区| 日韩三级免费观看| 7777精品伊人久久久大香线蕉的| 91福利在线播放| 99久久伊人精品| 大胆亚洲人体视频| 成人午夜大片免费观看| 国模套图日韩精品一区二区| 日本aⅴ亚洲精品中文乱码| 日韩av在线发布| 三级在线观看一区二区| 亚洲成av人片在线观看| 亚洲一区二区三区免费视频| 亚洲激情在线播放| 亚洲综合丝袜美腿| 亚洲电影视频在线| 日韩中文字幕麻豆| 久久精品国产一区二区| 老司机精品视频一区二区三区| 美女精品一区二区| 国产精品一区二区三区网站| 国产一区二区三区黄视频 | av一区二区三区在线| 成人激情免费电影网址| 91在线观看美女| 欧美在线999| 欧美久久高跟鞋激| 日韩欧美一区二区在线视频| 精品99一区二区| 国产午夜一区二区三区| 亚洲人成精品久久久久久 | 欧美猛男gaygay网站| 欧美一级日韩一级| 精品国产制服丝袜高跟| 国产欧美日韩精品在线| 亚洲图片激情小说| 午夜激情综合网| 国产麻豆一精品一av一免费| 懂色av一区二区三区蜜臀| 91在线视频播放地址| 欧美日韩久久久一区| 精品国产三级电影在线观看| 国产精品成人一区二区艾草| 亚洲一区二区av电影| 青青草97国产精品免费观看无弹窗版 | 亚洲地区一二三色| 久久精品国产99| 成人性生交大片免费看视频在线| 色综合久久天天| 日韩无一区二区| 中文字幕一区在线| 日日夜夜精品视频免费| 成人激情校园春色| 欧美一区二区三级| 亚洲国产精品激情在线观看| 偷拍亚洲欧洲综合| 成人的网站免费观看| 欧美一级夜夜爽| 亚洲天天做日日做天天谢日日欢 | 狠狠久久亚洲欧美| 91理论电影在线观看| 日韩一级欧美一级| 亚洲欧美成aⅴ人在线观看 | 日韩午夜在线影院| 亚洲三级小视频| 国产一区二区三区在线看麻豆 | 国产大陆a不卡| 欧美精品色一区二区三区| 国产欧美一区二区精品性色| 美女mm1313爽爽久久久蜜臀| 91免费视频网| 欧美国产亚洲另类动漫| 免费观看在线综合| 欧美日韩国产美女| 一区二区三区在线视频观看58| 国产一区二区日韩精品| 欧美一区二区三区视频| 亚洲一区二区在线观看视频| 不卡一二三区首页| 久久久影视传媒| 蜜乳av一区二区| 欧美日韩国产中文| 亚洲天堂成人在线观看| 国产精品亚洲成人| 久久嫩草精品久久久精品一| 免费黄网站欧美| 91超碰这里只有精品国产| 亚洲一级电影视频| 欧美午夜宅男影院| 亚洲一区二区在线播放相泽| 在线观看日韩电影| 亚洲欧美乱综合| 色综合久久久网| 一区二区三区中文在线| 99re成人精品视频| 国产精品网站一区| 91首页免费视频| 亚洲一区二区三区自拍|