亚洲欧美第一页_禁久久精品乱码_粉嫩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一区二区三区免费野_久草精品视频
午夜影院久久久| 91同城在线观看| 91国产视频在线观看| 欧美一区二区三区免费观看视频 | 精品一区二区三区在线观看国产| bt7086福利一区国产| 26uuu精品一区二区| 亚洲成a人在线观看| 91视频国产资源| 久久久久久久久久久久电影| 亚洲第一成年网| 91丝袜国产在线播放| 欧美激情在线免费观看| 激情小说亚洲一区| 欧美成人综合网站| 五月天丁香久久| 欧美人与性动xxxx| 五月开心婷婷久久| 欧美午夜精品久久久| 亚洲九九爱视频| 色综合色狠狠天天综合色| 欧美国产成人在线| 国产麻豆视频精品| 久久综合一区二区| 精品一区二区精品| 日韩欧美国产麻豆| 麻豆精品一区二区av白丝在线| 欧美日韩亚洲综合一区| 亚洲国产日韩a在线播放| 欧美亚洲综合久久| 亚洲一级片在线观看| 欧美日韩一区二区在线视频| 亚洲综合免费观看高清完整版| 一本色道**综合亚洲精品蜜桃冫| 国产精品久久久久久一区二区三区 | 国产伦精品一区二区三区免费迷 | 91精品国产一区二区三区蜜臀 | 欧美成人性战久久| 精品一二三四在线| 国产日韩成人精品| 成人午夜精品在线| 亚洲欧洲综合另类| 欧美日韩一本到| 久久国产精品无码网站| 精品久久国产97色综合| 国产成人自拍网| 中文字幕一区二区三区在线不卡| 色网站国产精品| 亚洲午夜日本在线观看| 欧美一区二区三区公司| 国产精品一级黄| 中文字幕巨乱亚洲| 欧美日韩一区二区三区四区五区 | 91在线porny国产在线看| 一区二区三区在线观看动漫| 欧美一区二区三区在| 粉嫩av一区二区三区粉嫩 | 国产嫩草影院久久久久| 97国产精品videossex| 亚洲成人黄色小说| 久久久久久电影| 色婷婷一区二区三区四区| 免费人成精品欧美精品| 久久久精品欧美丰满| 91久久久免费一区二区| 日韩在线播放一区二区| 国产精品免费网站在线观看| 欧美性生活影院| 国产二区国产一区在线观看| 亚洲欧美日韩国产手机在线 | 欧美日韩视频在线观看一区二区三区 | 99久久亚洲一区二区三区青草| 午夜一区二区三区视频| 久久伊99综合婷婷久久伊| 91久久精品日日躁夜夜躁欧美| 麻豆精品在线观看| 亚洲女人的天堂| 欧美成人a∨高清免费观看| 99久久精品费精品国产一区二区| 日韩精品一二三区| ●精品国产综合乱码久久久久| 日韩欧美三级在线| 欧美在线免费观看视频| 国产成人久久精品77777最新版本| 夜夜精品视频一区二区| 欧美激情艳妇裸体舞| 日韩一区二区麻豆国产| 在线免费观看成人短视频| 国产不卡视频在线播放| 美腿丝袜亚洲三区| 亚洲va天堂va国产va久| 亚洲免费视频成人| 国产精品免费观看视频| 精品免费视频一区二区| 这里只有精品电影| 欧美性三三影院| 91在线精品一区二区三区| 国产精品99久久久久久久vr| 日本aⅴ亚洲精品中文乱码| 亚洲乱码中文字幕| 中文字幕一区二区三区不卡在线 | 精品国产不卡一区二区三区| 欧美一区二区网站| 欧美男女性生活在线直播观看| 色婷婷狠狠综合| 91一区二区在线| 99久久婷婷国产| 91视频免费看| 欧美性受xxxx| 欧美日韩国产a| 欧美二区乱c少妇| 91麻豆精品国产91| 日韩视频在线你懂得| 日韩一区二区电影在线| 日韩网站在线看片你懂的| 欧美理论片在线| 欧美一卡2卡3卡4卡| 精品久久久久久无| 久久久久久久久岛国免费| 久久久精品影视| 国产欧美日韩在线视频| 国产精品国产三级国产有无不卡 | 国产毛片精品国产一区二区三区| 国产一区二区在线看| 国产裸体歌舞团一区二区| 粉嫩13p一区二区三区| 成人aa视频在线观看| 91片在线免费观看| 精品视频在线免费| 日韩欧美久久久| 国产欧美1区2区3区| 国产精品福利一区| 亚洲精品你懂的| 欧美aaaaa成人免费观看视频| 看电视剧不卡顿的网站| 粉嫩av一区二区三区| 日本电影欧美片| 欧美岛国在线观看| 亚洲国产电影在线观看| 亚洲精品中文在线影院| 性做久久久久久免费观看欧美| 久久电影网电视剧免费观看| 丁香六月综合激情| 欧美欧美欧美欧美| 国产欧美一区二区精品久导航 | 日韩激情在线观看| 国产精品一级片| 91传媒视频在线播放| 欧美电影免费观看完整版| 欧美国产欧美亚州国产日韩mv天天看完整| 国产精品久久久久久久岛一牛影视| 亚洲人成影院在线观看| 青娱乐精品视频| 成人av电影免费观看| 91精品国产欧美一区二区成人| 国产精品国产自产拍高清av| 午夜精品影院在线观看| 成人黄色大片在线观看| 91精品国产综合久久香蕉的特点| 欧美激情一二三区| 美女网站视频久久| 91国偷自产一区二区三区观看 | 一区二区成人在线视频| 国产一区二区在线观看免费| 欧美最新大片在线看| 国产欧美日韩卡一| 奇米777欧美一区二区| 91丨九色丨蝌蚪富婆spa| 亚洲精品在线免费观看视频| 亚洲一区二区三区国产| 成人综合在线视频| 久久婷婷综合激情| 免费观看一级欧美片| 欧美优质美女网站| 亚洲女与黑人做爰| 成人一级黄色片| 国产日韩精品一区| 国产在线不卡一区| 日韩精品一区二区三区四区| 天天做天天摸天天爽国产一区| av在线一区二区| 欧美激情综合五月色丁香小说| 久久av资源网| 日韩视频一区在线观看| 日韩国产欧美在线播放| 欧美亚洲国产bt| 亚洲自拍都市欧美小说| 不卡区在线中文字幕| 国产视频一区不卡| 国产成人在线影院| 国产欧美一区二区三区在线看蜜臀 | 精品一区二区三区在线观看| 久久久久久久久久久电影| 美女国产一区二区| 欧美www视频| 国产精品一二三| 国产婷婷精品av在线| 国产精品亚洲成人| 欧美国产欧美综合| proumb性欧美在线观看|