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

? 歡迎來(lái)到蟲(chóng)蟲(chóng)下載站! | ?? 資源下載 ?? 資源專(zhuān)輯 ?? 關(guān)于我們
? 蟲(chóng)蟲(chóng)下載站

?? sa1100.c

?? Linux2.4.20針對(duì)三星公司的s3c2440內(nèi)核基礎(chǔ)上的一些設(shè)備驅(qū)動(dòng)代碼
?? C
?? 第 1 頁(yè) / 共 4 頁(yè)
字號(hào):
/* * linux/drivers/usbd/sa1100_bi/udc.c * * Copyright (c) 2000, 2001, 2002 Lineo * Copyright (c) 2001 Hewlett Packard * * By:  *      Stuart Lynne <sl@lineo.com>,  *      Tom Rushworth <tbr@lineo.com>,  *      Bruce Balden <balden@lineo.com> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. * *//* * Please note that this driver works reasonably well with StrongARM parts * running at 206Mhz.  * * It is not possible to run the USB with DMA on the StrongARM running at 133Mhz. * Running without DMA is possible and reasonably reliable. This can be done by * restricting the USB packetsize to 16bytes for your bulk endpoints. */ #include <linux/config.h>#include <linux/module.h>#include "../usbd-export.h"#include "../usbd-build.h"#include "../usbd-module.h"MODULE_AUTHOR ("sl@lineo.com, tbr@lineo.com");MODULE_DESCRIPTION ("USB Device SA-1100 Bus Interface");USBD_MODULE_INFO ("sa1100_bi 0.2");#include <linux/kernel.h>#include <linux/slab.h>#include <linux/interrupt.h>#include <linux/pci.h>#include <linux/init.h>#include <linux/delay.h>#include <asm/atomic.h>#include <asm/io.h>#include <linux/proc_fs.h>#if LINUX_VERSION_CODE >= KERNEL_VERSION(2,4,6)#define USE_ADD_DEL_TIMER_FOR_USBADDR_CHECK 1#include <linux/timer.h>#else#undef USE_ADD_DEL_TIMER_FOR_USBADDR_CHECK#include <linux/tqueue.h>#endif#include <linux/netdevice.h>#include <linux/version.h>#include <linux/pci.h>#include <linux/cache.h>#include <asm/dma.h>#include <asm/mach/dma.h>#include <asm/irq.h>#include <asm/system.h>#include <asm/hardware.h>#include <asm/types.h>#include <asm/uaccess.h>#include <asm/io.h>#include <asm/pgtable.h>#include <asm/pgalloc.h>#include "../usbd-debug.h"#include "../usbd.h"#include "../usbd-func.h"#include "../usbd-bus.h"#include "../usbd-inline.h"#include "usbd-bi.h"#include "sa1100.h"#if defined(CONFIG_SA1100_COLLIE)       // XXX change to 5500#include <asm-arm/arch-sa1100/collie.h>#include <asm/ucb1200.h>#include <asm/arch/tc35143.h>#endif#define MIN(a,b) ((a) < (b) ? (a) : (b))#define MAX(a,b) ((a) > (b) ? (a) : (b))#define ACTIVEA 1#define ACTIVEB 2static struct usb_device_instance *udc_device;	// required for the interrupt handler/* * ep_endpoints - map physical endpoints to logical endpoints */static struct usb_endpoint_instance *ep_endpoints[UDC_MAX_ENDPOINTS];//static struct urb ep0_urb;unsigned char usb_address;extern unsigned int udc_interrupts;extern unsigned int udc_interrupts_last;unsigned int ep0_interrupts;unsigned int tx_interrupts;unsigned int rx_interrupts;unsigned int sus_interrupts;unsigned int res_interrupts;unsigned int udc_address_errors;unsigned int udc_ticks;unsigned int udc_fixed;unsigned int ep0_interrupts_last;unsigned int rx_interrupts_last;unsigned int tx_interrupts_last;unsigned int udc_rpe_errors;unsigned int udc_ep1_errors;unsigned int udc_ep2_errors;unsigned int udc_ep2_tpe;unsigned int udc_ep2_tur;unsigned int udc_ep2_sst;unsigned int udc_ep2_fst;int usbd_rcv_dma;int usbd_tx_dma;static int udc_saw_sof;#ifdef USE_ADD_DEL_TIMER_FOR_USBADDR_CHECKstatic struct timer_list sa1100_usb_dev_addr_check;static int usb_addr_check_initialized = 0;#define CHECK_INTERVAL   1#elsestatic struct tq_struct sa1100_tq;#endif/* * DMA control register structure */typedef struct {	volatile u_long DDAR;	volatile u_long SetDCSR;	volatile u_long ClrDCSR;	volatile u_long RdDCSR;	volatile dma_addr_t DBSA;	volatile u_long DBTA;	volatile dma_addr_t DBSB;	volatile u_long DBTB;} dma_regs_t;/* ********************************************************************************************* *//* IO */volatile int udc (volatile unsigned int *regaddr){	volatile unsigned int value;	int ok;	for (ok = 1000, value = *(regaddr); value != *(regaddr) && ok--; value = *(regaddr));	if (!ok) {		dbg_udc (0, "NOT OK: %p %x", regaddr, value);	}	return value;}static __inline__ volatile int _udc (volatile unsigned int *regaddr){	volatile unsigned int value;	int ok;	for (ok = 1000, value = *(regaddr); value != *(regaddr) && ok--; value = *(regaddr));	if (!ok) {		printk (KERN_ERR "NOT OK: %p %x\n", regaddr, value);	}	return value;}static __inline__ void _sa1100_bi_dma_run (dmach_t channel, int active){	dma_regs_t *regs = (dma_regs_t *) io_p2v (_DDAR (channel));	if (active == ACTIVEA) {		regs->SetDCSR = DCSR_STRTA | DCSR_RUN;	} else {		regs->SetDCSR = DCSR_STRTB | DCSR_RUN;	}}static __inline__ int _sa1100_bi_dma_queue_buffer_irq (dmach_t channel, dma_addr_t data, int size){	int status;	dma_regs_t *regs = (dma_regs_t *) io_p2v (_DDAR (channel));	status = regs->RdDCSR;	if (((status & DCSR_BIU) && (status & DCSR_STRTB)) || (!(status & DCSR_BIU) && !(status & DCSR_STRTA))) {		regs->ClrDCSR = DCSR_DONEA | DCSR_STRTA;		regs->DBSA = data;		regs->DBTA = size;		// Once is good, twice is better....		regs->DBSA = data;		regs->DBTA = size;		return ACTIVEA;	} else {		regs->ClrDCSR = DCSR_DONEB | DCSR_STRTB;		regs->DBSB = data;		regs->DBTB = size;		// Once is good, twice is better....		regs->DBSB = data;		regs->DBTB = size;		return ACTIVEB;	}}static __inline__ int _sa1100_bi_dma_flush_all_irq (dmach_t channel){	dma_regs_t *regs = (dma_regs_t *) io_p2v (_DDAR (channel));	regs->ClrDCSR = DCSR_STRTA | DCSR_STRTB | DCSR_RUN | DCSR_IE;	return 0;}static __inline__ int _sa1100_bi_dma_stop_get_current_irq (dmach_t channel, dma_addr_t * addr, int active){	dma_regs_t *regs = (dma_regs_t *) io_p2v (_DDAR (channel));	// addr sometimes can be set incorrectly, the caller must do bounds checking	switch (active) {	case ACTIVEA:		regs->ClrDCSR = DCSR_RUN | DCSR_STRTA | DCSR_RUN | DCSR_IE;		*addr = regs->DBSA;	// not reliable		return 0;	case ACTIVEB:		regs->ClrDCSR = DCSR_RUN | DCSR_STRTB | DCSR_RUN | DCSR_IE;		*addr = regs->DBSB;	// not reliable		return 0;	default:		*addr = 0;		return -ENXIO;	}}/* ********************************************************************************************* */static u32 getCPUID (char **stepping){	u32 cpuID;	__asm__ __volatile__ (" mrc   p15, 0, %0, c0, c0":"=r" (cpuID)			      :);	if (NULL != stepping) {		switch (cpuID & 0xf) {		case 0:			*stepping = "A0";			break;		case 4:			*stepping = "B0";			break;		case 5:			*stepping = "B1";			break;		case 6:			*stepping = "B2";			break;		case 8:			*stepping = "B4";			break;		case 9:			*stepping = "B5";			break;		default:			*stepping = "??";			dbg_udc (0, "stepping unknown, ID#%x", (cpuID & 0xf));		}	}	return (cpuID);}void udc_fix_errata_29 (char *msg){	int ok;	u32 cpuID;		// from init.c	cpuID = getCPUID (NULL);	// Set errata 29 fix bit, if possible	if (cpuID == 0 || (cpuID & 0xF) >= 9) {		// Unknown CPU, or B5 stepping and above		// set errata 29 fix enable (for B5 and above, B4 will ignore)		int udc_cr;		for (ok = 10; ok > 0; ok--) {			*(UDCCR) |= /*UDCCR_ERR29 */ 0x80;			// Do some dummy reads....			udc_cr = *(UDCCR);			udc_cr = *(UDCCR);			udc_cr = *(UDCCR);			if (udc (UDCCR) & /*UDCCR_ERR29 */ 0x80) {				dbg_udc (0,					 "%s: set errata 29 fix bit worked, UDCCR#%02x ok=%d cpuID#%08x",					 msg, udc (UDCCR), ok, cpuID);				ok = -2;				break;			}		}		if (ok != -2) {			dbg_udc (0, "%s: set errata 29 fix bit failed, UDCCR#%02x ok=%d cpuID#%08x",				 msg, udc (UDCCR), ok, cpuID);		}	} else {		dbg_udc (0, "%s: errata 29 fix bit not available, cpuID#%08x", msg, cpuID);	}}/* ********************************************************************************************* *//** * sa1100_tick - clock timer task  * @data: *   * Run from global clock tick to check if we are suspended. */#ifdef USE_ADD_DEL_TIMER_FOR_USBADDR_CHECKstatic void sa1100_tick (unsigned long data)#elsestatic void sa1100_tick (void *data)#endif{	udc_ticks++;	// is driver active	if (data) {		if (_udc (UDCAR) != usb_address) {			dbg_udc (0, "sa1100_tick: ADDRESS ERROR DETECTED %02x %02x", *(UDCAR), usb_address);			udc_address_errors++;			udc_fixed++;		}		*(UDCAR) = usb_address;		// re-queue task#ifdef USE_ADD_DEL_TIMER_FOR_USBADDR_CHECK		sa1100_usb_dev_addr_check.expires = jiffies + CHECK_INTERVAL;		add_timer (&sa1100_usb_dev_addr_check);#else		queue_task (&sa1100_tq, &tq_timer);#endif	}}/* ********************************************************************************************* *//* Interrupt Handler *//** * int_hndlr_cable - interrupt handler for cable */static void int_hndlr_cable (int irq, void *dev_id, struct pt_regs *regs){#ifdef CONFIG_SA1100_USBCABLE_GPIO	dbg_udc (1, "udc_cradle_interrupt:");	udc_cable_event ();#endif}/** * int_hndlr_device - interrupt handler * */static void int_hndlr_device (int irq, void *dev_id, struct pt_regs *regs){	unsigned int status;	status = *(UDCSR);	*(UDCSR) = status;	udc_interrupts++;	//dbg_udc(0, "");	dbg_intr(2, "[%d]: CSR: %02x CCR: %02x CAR: %02x:%02x", udc_interrupts, status, *(UDCCR), *(UDCAR), usb_address);	// Handle common interrupts first, IN (tx) and OUT (recv)	if (status & UDCSR_RIR) {		ep1_int_hndlr (status);	}	if (status & UDCSR_TIR) {		ep2_int_hndlr (status, 1);	}	// handle less common interrupts	if (status & (UDCSR_EIR | UDCSR_RSTIR | UDCSR_SUSIR | UDCSR_RESIR)) {		if (status & UDCSR_EIR) {			ep0_int_hndlr (status);		}		if (status & UDCSR_RSTIR) {			dbg_intr (1, "[%d] DEVICE_RESET: CSR: %02x CS0: %02x CAR: %02x",                                         udc_interrupts, *(UDCSR), *(UDCCS0), *(UDCAR));			usbd_device_event (udc_device, DEVICE_RESET, 0);		}		if (status & UDCSR_SUSIR) {			dbg_intr (1, "[%d] SUSPEND address: %02x irq: %02x status: %02x",                                         udc_interrupts, *(UDCAR), irq, status);			sus_interrupts++;#if defined(CONFIG_SA1100_COLLIE)	// XXX change to 5500			usbd_device_event (udc_device, DEVICE_BUS_INACTIVE, 0);#else			usbd_device_event (udc_device, DEVICE_BUS_INACTIVE, 0);			//usbd_device_event (udc_device, DEVICE_RESET, 0);#endif			udc_suspended_interrupts (udc_device);			udc_ticker_poke ();		}		if (status & UDCSR_RESIR) {			dbg_intr (1, "[%d] RESUME address: %02x irq: %02x status: %02x", udc_interrupts, *(UDCAR), irq, status);                        *(UDCAR) = usb_address;			res_interrupts++;#if defined(CONFIG_SA1100_COLLIE)	// XXX change to 5500			usbd_device_event (udc_device, DEVICE_BUS_ACTIVITY, 0);#endif			udc_all_interrupts (udc_device);			udc_ticker_poke ();		}	}	// Check that the UDC has not forgotton it's address, force it back to correct value	//if (_udc (UDCAR) != usb_address) {	//	udc_address_errors++;	//}	*(UDCAR) = usb_address;}/* ********************************************************************************************* *//* * Start of public functions. *//** * udc_init - initialize * * Return non-zero if we cannot see device. **/int udc_init (void){	// reset	return 0;}/** * udc_start_in_irq - start transmit * @endpoint: * * Called with interrupts disabled. */void udc_start_in_irq (struct usb_endpoint_instance *endpoint){	ep2_int_hndlr (0, 0);}void udc_stall_ep0 (void){	int ok = 0;	// QQQ eh?	dbg_udc (0, "stalling ep0 (UDCCS0_FST,UDCCS0_FST,UDCCS0_SST)");	// write 1 to set FST	SET_AND_TEST ((*(UDCCS0) &= UDCCS0_FST), !(udc (UDCCS0) & UDCCS0_FST), ok);	if (!ok) {		dbg_udc (0, "cannot stall !(UDCCS0&UDCCS0_FST) UDCCS0: %02x", *(UDCCS0));	}	// write 0 to reset FST	SET_AND_TEST ((*(UDCCS0) &= ~UDCCS0_FST), (udc (UDCCS0) & UDCCS0_FST), ok);	if (!ok) {		dbg_udc (0, "cannot stall (UDCCS0&UDCCS0_FST) UDCCS0: %02x", *(UDCCS0));	}	// write 1 to reset SST	SET_AND_TEST ((*(UDCCS0) = UDCCS0_SST), (udc (UDCCS0) & UDCCS0_SST), ok);	if (!ok) {		dbg_udc (0, "cannot stall (UDCCS0&UDCCS0_SST) UDCCS0: %02x", *(UDCCS0));	}}static void stall_ep_n (int ep, volatile unsigned int *regaddr, int fst){	int ok;	dbg_udc (0, "stalling ep %d (FST)", ep);	// write 1 to set FST	SET_AND_TEST ((*(regaddr) = fst), !(udc (regaddr) & fst), ok);	if (!ok) {		dbg_udc (0, "cannot stall !(reg&fst) UDCCS%d: %02x", ep, *(regaddr));	}}static void reset_ep_n (int ep, volatile unsigned int *regaddr, int fst, int sst){	int ok;	dbg_udc (1, "reset ep %d (FST)", ep);	// write 0 to reset FST	SET_AND_TEST ((*(regaddr) &= ~fst), (udc (regaddr) & fst), ok);	if (!ok) {		dbg_udc (0, "cannot stall !(reg&fst) UDCCS%d: %02x", ep, *(regaddr));	}	// write 1 to reset SST	SET_AND_TEST ((*(regaddr) = sst), (udc (regaddr) & sst), ok);	if (!ok) {		dbg_udc (0, "cannot stall (reg&sst) UDCCS%d: %02x", ep, *(UDCCS0));	}}/** * udc_stall_ep - stall endpoint * @ep: physical endpoint * * Stall the endpoint. */void udc_stall_ep (unsigned int ep){	dbg_udc (0, "STALLING %d (FST)", ep);	switch (ep) {	case 1:		stall_ep_n (1, UDCCS1, UDCCS1_FST);		break;	case 2:		stall_ep_n (2, UDCCS2, UDCCS2_FST);		break;	}}/** * udc_reset_ep - reset endpoint * @ep: physical endpoint * reset the endpoint. * * returns : 0 if ok, -1 otherwise */void udc_reset_ep (unsigned int ep){	dbg_udc (1, "RESETING %d (FST)", ep);	switch (ep) {	case 0:		reset_ep_n (1, UDCCS0, UDCCS0_FST, UDCCS0_SST);		break;	case 1:		reset_ep_n (1, UDCCS1, UDCCS1_FST, UDCCS1_SST);		break;	case 2:		reset_ep_n (2, UDCCS2, UDCCS2_FST, UDCCS2_SST);		break;	}}

?? 快捷鍵說(shuō)明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號(hào) Ctrl + =
減小字號(hào) Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
99久久99久久免费精品蜜臀| 欧美一区二区二区| 成人一级黄色片| 成人白浆超碰人人人人| 懂色av一区二区夜夜嗨| 国产不卡视频一区二区三区| 粉嫩嫩av羞羞动漫久久久| 成人中文字幕在线| 波多野洁衣一区| 99久久精品国产网站| 一本久久a久久精品亚洲| 91福利视频网站| 欧美日韩一二三| 欧美丰满美乳xxx高潮www| 欧美丰满一区二区免费视频| 在线综合视频播放| 精品欧美久久久| 欧美激情一区二区在线| 中文字幕亚洲一区二区va在线| 国产精品动漫网站| 亚洲黄一区二区三区| 亚洲成人免费视频| 麻豆91在线观看| 国产一区二三区好的| 成人午夜激情视频| 91老师国产黑色丝袜在线| 欧美性大战久久久久久久| 欧美精品 日韩| 久久久久久日产精品| 亚洲欧美综合色| 亚洲成精国产精品女| 美国一区二区三区在线播放| 黄网站免费久久| youjizz久久| 欧美日韩国产一二三| 欧美www视频| **欧美大码日韩| 亚洲国产成人av| 国产一区高清在线| 91久久国产最好的精华液| 日韩一卡二卡三卡| 国产精品久久久久国产精品日日| 一区二区不卡在线视频 午夜欧美不卡在| 午夜精品福利视频网站| 韩国女主播成人在线| 91亚洲国产成人精品一区二三| 欧美日韩激情一区| 久久久国产午夜精品| 一区二区三区四区不卡在线 | 欧美日韩一区二区三区不卡| 欧美一级片在线| 国产精品入口麻豆原神| 亚洲国产精品嫩草影院| 国产老妇另类xxxxx| 在线国产电影不卡| 精品久久久久久久久久久久久久久久久 | 在线看一区二区| 欧美成人三级电影在线| 亚洲视频一区二区在线| 国产又黄又大久久| 精品视频999| 国产精品成人免费| 久久99国产乱子伦精品免费| 日本道在线观看一区二区| 久久丝袜美腿综合| 亚洲午夜私人影院| 岛国精品在线观看| 日韩欧美国产一区二区三区| 亚洲美女淫视频| 国产成人免费视频| 日韩视频在线永久播放| 亚洲在线视频网站| www.成人网.com| 精品国产乱码91久久久久久网站| 一区二区三区在线视频免费 | 91成人免费网站| 久久精品欧美日韩| 蜜臀国产一区二区三区在线播放| 一本色道久久综合亚洲aⅴ蜜桃| 国产日韩精品久久久| 精品在线播放午夜| 欧美一级片免费看| 视频一区视频二区中文| 在线观看视频91| 1区2区3区精品视频| 激情深爱一区二区| 欧美刺激午夜性久久久久久久 | 成人av在线影院| 精品国产精品网麻豆系列| 日韩二区在线观看| 欧美精品三级日韩久久| 亚洲国产一二三| 色欧美日韩亚洲| 亚洲欧洲色图综合| av电影一区二区| 国产精品免费aⅴ片在线观看| 精品一区二区在线免费观看| 日韩一级成人av| 日本在线观看不卡视频| 制服丝袜国产精品| 热久久免费视频| 欧美一区二区三区电影| 午夜激情综合网| 在线综合+亚洲+欧美中文字幕| 亚洲成人黄色小说| 欧美日本一区二区| 日本中文字幕一区| 欧美一区在线视频| 裸体一区二区三区| 欧美成人乱码一区二区三区| 老司机免费视频一区二区| 欧美电影免费观看完整版| 美国十次综合导航| 2017欧美狠狠色| 国产99久久久国产精品潘金 | 国产精品久久久久久一区二区三区| 国产精品88888| 中文字幕成人在线观看| 成人精品视频网站| 亚洲另类中文字| 欧美色精品天天在线观看视频| 视频在线观看国产精品| 精品少妇一区二区三区| 国产又粗又猛又爽又黄91精品| 亚洲国产精品99久久久久久久久| 成人不卡免费av| 亚洲一区二区三区爽爽爽爽爽| 在线成人午夜影院| 国产美女娇喘av呻吟久久| 国产精品对白交换视频| 欧美性猛片aaaaaaa做受| 婷婷久久综合九色综合绿巨人| 日韩精品一区二区三区在线播放 | 成人免费视频网站在线观看| 国产精品福利av| 欧美乱妇23p| 国产在线一区二区| 亚洲乱码国产乱码精品精的特点| 欧美一区二区三区在线看| 国产在线精品视频| 中文字幕综合网| 91精品在线免费观看| 国产福利不卡视频| 亚洲国产日日夜夜| 久久精品在线免费观看| 色系网站成人免费| 久久丁香综合五月国产三级网站| 国产精品久久久久久久第一福利| 欧美日韩国产免费一区二区| 国产精品一区二区在线播放| 亚洲九九爱视频| 精品国产髙清在线看国产毛片| 91啪亚洲精品| 久草精品在线观看| 亚洲精品免费在线播放| 精品日韩99亚洲| 欧美综合天天夜夜久久| 国产盗摄精品一区二区三区在线| 亚洲综合色在线| 久久精品欧美一区二区三区不卡| 欧美亚洲动漫精品| 国产成人综合亚洲91猫咪| 午夜婷婷国产麻豆精品| 国产精品乱码久久久久久| 4438x亚洲最大成人网| av高清不卡在线| 国精产品一区一区三区mba桃花| 亚洲综合一区在线| 久久久久久久久久看片| 欧美绝品在线观看成人午夜影视| 成人免费福利片| 韩国av一区二区三区在线观看| 亚洲一区二区黄色| 国产精品国产三级国产普通话蜜臀| 日韩亚洲欧美成人一区| 欧美三级日韩三级国产三级| 99久久精品99国产精品| 国产乱子轮精品视频| 另类的小说在线视频另类成人小视频在线 | 波多野结衣中文一区| 久久精品国产免费| 亚洲成va人在线观看| 亚洲另类色综合网站| 欧美激情中文不卡| 精品成人在线观看| 8x8x8国产精品| 欧美在线视频全部完| 91丨porny丨户外露出| 国产69精品久久777的优势| 免费在线一区观看| 日日摸夜夜添夜夜添精品视频 | 极品瑜伽女神91| 日韩 欧美一区二区三区| 亚洲一区二区黄色| 亚洲综合成人网| 一区二区三区在线视频免费 | 99久久精品国产导航| 成人激情动漫在线观看| 国产成人免费视频网站 | 北岛玲一区二区三区四区|