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

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關(guān)于我們
? 蟲蟲下載站

?? hid-tmff.c

?? 優(yōu)龍2410linux2.6.8內(nèi)核源代碼
?? C
字號(hào):
/* * Force feedback support for various HID compliant devices by ThrustMaster: *    ThrustMaster FireStorm Dual Power 2 * and possibly others whose device ids haven't been added. * *  Modified to support ThrustMaster devices by Zinx Verituse *  on 2003-01-25 from the Logitech force feedback driver, *  which is by Johann Deneux. * *  Copyright (c) 2003 Zinx Verituse <zinx@epicsol.org> *  Copyright (c) 2002 Johann Deneux *//* * 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */#include <linux/input.h>#include <linux/sched.h>#undef DEBUG#include <linux/usb.h>#include <linux/circ_buf.h>#include "hid.h"#include "fixp-arith.h"/* Usages for thrustmaster devices I know about */#define THRUSTMASTER_USAGE_RUMBLE_LR	(HID_UP_GENDESK | 0xbb)#define DELAY_CALC(t,delay)		((t) + (delay)*HZ/1000)/* Effect status */#define EFFECT_STARTED 0	/* Effect is going to play after some time */#define EFFECT_PLAYING 1	/* Effect is playing */#define EFFECT_USED    2/* For tmff_device::flags */#define DEVICE_CLOSING 0	/* The driver is being unitialised *//* Check that the current process can access an effect */#define CHECK_OWNERSHIP(effect) (current->pid == 0 \        || effect.owner == current->pid)#define TMFF_CHECK_ID(id)	((id) >= 0 && (id) < TMFF_EFFECTS)#define TMFF_CHECK_OWNERSHIP(i, l) \        (test_bit(EFFECT_USED, l->effects[i].flags) \        && CHECK_OWNERSHIP(l->effects[i]))#define TMFF_EFFECTS 8struct tmff_effect {	pid_t owner;	struct ff_effect effect;	unsigned long flags[1];	unsigned int count;             /* Number of times left to play */	unsigned long play_at;          /* When the effect starts to play */	unsigned long stop_at;		/* When the effect ends */};struct tmff_device {	struct hid_device *hid;	struct hid_report *report;	struct hid_field *rumble;	unsigned int effects_playing;	struct tmff_effect effects[TMFF_EFFECTS];	spinlock_t lock;             /* device-level lock. Having locks on					a per-effect basis could be nice, but					isn't really necessary */	unsigned long flags[1];      /* Contains various information about the					state of the driver for this device */	struct timer_list timer;};/* Callbacks */static void hid_tmff_exit(struct hid_device *hid);static int hid_tmff_event(struct hid_device *hid, struct input_dev *input,			  unsigned int type, unsigned int code, int value);static int hid_tmff_flush(struct input_dev *input, struct file *file);static int hid_tmff_upload_effect(struct input_dev *input,				  struct ff_effect *effect);static int hid_tmff_erase(struct input_dev *input, int id);/* Local functions */static void hid_tmff_recalculate_timer(struct tmff_device *tmff);static void hid_tmff_timer(unsigned long timer_data);int hid_tmff_init(struct hid_device *hid){	struct tmff_device *private;	struct list_head *pos;	struct hid_input *hidinput = list_entry(hid->inputs.next, struct hid_input, list);	private = kmalloc(sizeof(struct tmff_device), GFP_KERNEL);	if (!private)		return -ENOMEM;	memset(private, 0, sizeof(struct tmff_device));	hid->ff_private = private;	/* Find the report to use */	__list_for_each(pos, &hid->report_enum[HID_OUTPUT_REPORT].report_list) {		struct hid_report *report = (struct hid_report *)pos;		int fieldnum;		for (fieldnum = 0; fieldnum < report->maxfield; ++fieldnum) {			struct hid_field *field = report->field[fieldnum];			if (field->maxusage <= 0)				continue;			switch (field->usage[0].hid) {				case THRUSTMASTER_USAGE_RUMBLE_LR:					if (field->report_count < 2) {						warn("ignoring THRUSTMASTER_USAGE_RUMBLE_LR with report_count < 2");						continue;					}					if (field->logical_maximum == field->logical_minimum) {						warn("ignoring THRUSTMASTER_USAGE_RUMBLE_LR with logical_maximum == logical_minimum");						continue;					}					if (private->report && private->report != report) {						warn("ignoring THRUSTMASTER_USAGE_RUMBLE_LR in other report");						continue;					}					if (private->rumble && private->rumble != field) {						warn("ignoring duplicate THRUSTMASTER_USAGE_RUMBLE_LR");						continue;					}					private->report = report;					private->rumble = field;					set_bit(FF_RUMBLE, hidinput->input.ffbit);					break;				default:					warn("ignoring unknown output usage %08x", field->usage[0].hid);					continue;			}			/* Fallthrough to here only when a valid usage is found */			hidinput->input.upload_effect = hid_tmff_upload_effect;			hidinput->input.flush = hid_tmff_flush;			set_bit(EV_FF, hidinput->input.evbit);			hidinput->input.ff_effects_max = TMFF_EFFECTS;		}	}	private->hid = hid;	spin_lock_init(&private->lock);	init_timer(&private->timer);	private->timer.data = (unsigned long)private;	private->timer.function = hid_tmff_timer;	/* Event and exit callbacks */	hid->ff_exit = hid_tmff_exit;	hid->ff_event = hid_tmff_event;	info("Force feedback for ThrustMaster rumble pad devices by Zinx Verituse <zinx@epicsol.org>");	return 0;}static void hid_tmff_exit(struct hid_device *hid){	struct tmff_device *tmff = hid->ff_private;	unsigned long flags;	spin_lock_irqsave(&tmff->lock, flags);	set_bit(DEVICE_CLOSING, tmff->flags);	del_timer_sync(&tmff->timer);	spin_unlock_irqrestore(&tmff->lock, flags);	kfree(tmff);}static int hid_tmff_event(struct hid_device *hid, struct input_dev *input,			  unsigned int type, unsigned int code, int value){	struct tmff_device *tmff = hid->ff_private;	struct tmff_effect *effect = &tmff->effects[code];	unsigned long flags;	if (type != EV_FF)		return -EINVAL;	if (!TMFF_CHECK_ID(code))		return -EINVAL;	if (!TMFF_CHECK_OWNERSHIP(code, tmff))		return -EACCES;	if (value < 0)		return -EINVAL;	spin_lock_irqsave(&tmff->lock, flags);	if (value > 0) {		set_bit(EFFECT_STARTED, effect->flags);		clear_bit(EFFECT_PLAYING, effect->flags);		effect->count = value;		effect->play_at = DELAY_CALC(jiffies, effect->effect.replay.delay);	} else {		clear_bit(EFFECT_STARTED, effect->flags);		clear_bit(EFFECT_PLAYING, effect->flags);	}	hid_tmff_recalculate_timer(tmff);	spin_unlock_irqrestore(&tmff->lock, flags);	return 0;}/* Erase all effects this process owns */static int hid_tmff_flush(struct input_dev *dev, struct file *file){	struct hid_device *hid = dev->private;	struct tmff_device *tmff = hid->ff_private;	int i;	for (i=0; i<dev->ff_effects_max; ++i)	     /* NOTE: no need to lock here. The only times EFFECT_USED is		modified is when effects are uploaded or when an effect is		erased. But a process cannot close its dev/input/eventX fd		and perform ioctls on the same fd all at the same time */		if (current->pid == tmff->effects[i].owner		     && test_bit(EFFECT_USED, tmff->effects[i].flags))			if (hid_tmff_erase(dev, i))				warn("erase effect %d failed", i);	return 0;}static int hid_tmff_erase(struct input_dev *dev, int id){	struct hid_device *hid = dev->private;	struct tmff_device *tmff = hid->ff_private;	unsigned long flags;	if (!TMFF_CHECK_ID(id))		return -EINVAL;	if (!TMFF_CHECK_OWNERSHIP(id, tmff))		return -EACCES;	spin_lock_irqsave(&tmff->lock, flags);	tmff->effects[id].flags[0] = 0;	hid_tmff_recalculate_timer(tmff);	spin_unlock_irqrestore(&tmff->lock, flags);	return 0;}static int hid_tmff_upload_effect(struct input_dev *input,				  struct ff_effect *effect){	struct hid_device *hid = input->private;	struct tmff_device *tmff = hid->ff_private;	int id;	unsigned long flags;	if (!test_bit(effect->type, input->ffbit))		return -EINVAL;	if (effect->id != -1 && !TMFF_CHECK_ID(effect->id))		return -EINVAL;	spin_lock_irqsave(&tmff->lock, flags);	if (effect->id == -1) {		/* Find a free effect */		for (id = 0; id < TMFF_EFFECTS && test_bit(EFFECT_USED, tmff->effects[id].flags); ++id);		if (id >= TMFF_EFFECTS) {			spin_unlock_irqrestore(&tmff->lock, flags);			return -ENOSPC;		}		effect->id = id;		tmff->effects[id].owner = current->pid;		tmff->effects[id].flags[0] = 0;		set_bit(EFFECT_USED, tmff->effects[id].flags);	} else {		/* Re-uploading an owned effect, to change parameters */		id = effect->id;		clear_bit(EFFECT_PLAYING, tmff->effects[id].flags);	}	tmff->effects[id].effect = *effect;	hid_tmff_recalculate_timer(tmff);	spin_unlock_irqrestore(&tmff->lock, flags);	return 0;}/* Start the timer for the next start/stop/delay *//* Always call this while tmff->lock is locked */static void hid_tmff_recalculate_timer(struct tmff_device *tmff){	int i;	int events = 0;	unsigned long next_time;	next_time = 0;	/* Shut up compiler's incorrect warning */	/* Find the next change in an effect's status */	for (i = 0; i < TMFF_EFFECTS; ++i) {		struct tmff_effect *effect = &tmff->effects[i];		unsigned long play_time;		if (!test_bit(EFFECT_STARTED, effect->flags))			continue;		effect->stop_at = DELAY_CALC(effect->play_at, effect->effect.replay.length);		if (!test_bit(EFFECT_PLAYING, effect->flags))			play_time = effect->play_at;		else			play_time = effect->stop_at;		events++;		if (time_after(jiffies, play_time))			play_time = jiffies;		if (events == 1)			next_time = play_time;		else {			if (time_after(next_time, play_time))				next_time = play_time;		}	}	if (!events && tmff->effects_playing) {		/* Treat all effects turning off as an event */		events = 1;		next_time = jiffies;	}	if (!events) {		/* No events, no time, no need for a timer. */		del_timer_sync(&tmff->timer);		return;	}	mod_timer(&tmff->timer, next_time);}/* Changes values from 0 to 0xffff into values from minimum to maximum */static inline int hid_tmff_scale(unsigned int in, int minimum, int maximum){	int ret;	ret = (in * (maximum - minimum) / 0xffff) + minimum;	if (ret < minimum)		return minimum;	if (ret > maximum)		return maximum;	return ret;}static void hid_tmff_timer(unsigned long timer_data){	struct tmff_device *tmff = (struct tmff_device *) timer_data;	struct hid_device *hid = tmff->hid;	unsigned long flags;	int left = 0, right = 0;	/* Rumbling */	int i;	spin_lock_irqsave(&tmff->lock, flags);	tmff->effects_playing = 0;	for (i = 0; i < TMFF_EFFECTS; ++i) {		struct tmff_effect *effect = &tmff->effects[i];		if (!test_bit(EFFECT_STARTED, effect->flags))			continue;		if (!time_after(jiffies, effect->play_at))			continue;		if (time_after(jiffies, effect->stop_at)) {			dbg("Finished playing once %d", i);			clear_bit(EFFECT_PLAYING, effect->flags);			if (--effect->count <= 0) {				dbg("Stopped %d", i);				clear_bit(EFFECT_STARTED, effect->flags);				continue;			} else {				dbg("Start again %d", i);				effect->play_at = DELAY_CALC(jiffies, effect->effect.replay.delay);				continue;			}		}		++tmff->effects_playing;		set_bit(EFFECT_PLAYING, effect->flags);		switch (effect->effect.type) {			case FF_RUMBLE:				right += effect->effect.u.rumble.strong_magnitude;				left += effect->effect.u.rumble.weak_magnitude;				break;			default:				BUG();				break;		}	}	left = hid_tmff_scale(left, tmff->rumble->logical_minimum, tmff->rumble->logical_maximum);	right = hid_tmff_scale(right, tmff->rumble->logical_minimum, tmff->rumble->logical_maximum);	if (left != tmff->rumble->value[0] || right != tmff->rumble->value[1]) {		tmff->rumble->value[0] = left;		tmff->rumble->value[1] = right;		dbg("(left,right)=(%08x, %08x)", left, right);		hid_submit_report(hid, tmff->report, USB_DIR_OUT);	}	if (!test_bit(DEVICE_CLOSING, tmff->flags))		hid_tmff_recalculate_timer(tmff);	spin_unlock_irqrestore(&tmff->lock, flags);}

?? 快捷鍵說明

復(fù)制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號(hào) Ctrl + =
減小字號(hào) Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
日韩一级欧美一级| 懂色av一区二区夜夜嗨| 欧洲精品一区二区| 一区二区三区精品视频| fc2成人免费人成在线观看播放| 欧美大度的电影原声| 免费成人av资源网| 精品久久久久久久久久久久久久久 | 亚洲视频你懂的| 不卡欧美aaaaa| 成人免费在线播放视频| 972aa.com艺术欧美| 亚洲精品v日韩精品| 欧美性做爰猛烈叫床潮| 男女男精品视频| 国产欧美精品区一区二区三区| 国产不卡视频在线观看| 中文字幕欧美一| 日韩一区二区三区观看| 久久精品国产77777蜜臀| 久久久久久亚洲综合影院红桃| 高清不卡在线观看av| 性做久久久久久久免费看| 精品国产伦一区二区三区免费| 成人三级伦理片| 日韩精品电影在线| 国产精品久久一级| 日韩一区二区免费电影| 成人v精品蜜桃久久一区| 日本中文字幕一区| 亚洲国产成人高清精品| 国产精品女上位| 久久久精品中文字幕麻豆发布| 欧美日韩精品一区二区三区蜜桃| 北条麻妃国产九九精品视频| 精品一区二区三区免费观看| 亚洲免费在线播放| 亚洲国产岛国毛片在线| 精品国产91九色蝌蚪| 欧美日韩在线综合| 欧美日韩一卡二卡三卡| 日本精品视频一区二区三区| 99精品欧美一区二区三区综合在线| 九色porny丨国产精品| 免费在线观看视频一区| 天堂在线亚洲视频| 亚洲综合丝袜美腿| 亚洲国产乱码最新视频| 日日夜夜免费精品| 三级成人在线视频| 国产在线精品免费av| 国产精品一区二区三区99| thepron国产精品| 色婷婷久久久久swag精品| 欧美吻胸吃奶大尺度电影| 欧美日韩国产大片| 日韩三级在线观看| 国产精品久久久久久久久动漫 | 中文字幕不卡在线播放| 亚洲主播在线观看| 国内外精品视频| 91老司机福利 在线| 日韩一区二区不卡| 国产精品区一区二区三| 五月综合激情日本mⅴ| 国产精品白丝av| 7777精品伊人久久久大香线蕉的 | 亚洲国产精品成人综合| 亚洲一区二区四区蜜桃| 国内精品视频666| 在线观看亚洲成人| 亚洲国产精品成人久久综合一区| 亚洲成av人片在线观看| 99视频精品在线| 久久久久久久久一| 天天做天天摸天天爽国产一区| 成人av电影在线| 久久中文字幕电影| 另类小说视频一区二区| 在线一区二区三区做爰视频网站| 日韩欧美一级二级三级久久久| 亚洲免费在线观看视频| 成人一级片网址| 欧美激情在线观看视频免费| 免费成人性网站| 欧美一级理论片| 视频一区免费在线观看| 欧美亚洲国产一卡| 午夜亚洲福利老司机| 色婷婷综合久久久中文字幕| 中文字幕一区二区三区四区 | 国产精品美女久久久久久久| 狠狠色狠狠色综合日日91app| 日韩亚洲欧美成人一区| 亚洲欧美电影一区二区| 日韩精品一卡二卡三卡四卡无卡| 色综合一区二区三区| 1024成人网色www| 成人免费高清视频在线观看| 国产精品欧美综合在线| 风流少妇一区二区| 久久婷婷久久一区二区三区| 国产剧情一区在线| 国产精品免费观看视频| 色哟哟一区二区在线观看| 亚洲精品福利视频网站| 欧美日韩一区二区三区视频| 日韩二区三区在线观看| 国产亚洲成av人在线观看导航| 国产成人精品免费| 亚洲成av人片在线| 日韩午夜在线观看| 99综合电影在线视频| 亚洲一级二级三级| www亚洲一区| 欧美日韩视频在线一区二区| 蜜臀久久久99精品久久久久久| 国产性天天综合网| 91精品国产综合久久福利软件| 国产中文字幕精品| 亚洲成精国产精品女| 国产欧美日韩精品一区| 欧美日韩一区二区三区在线看| 韩国三级在线一区| 天天综合色天天| 亚洲三级小视频| 中文字幕不卡在线观看| 欧美一级黄色录像| 欧美理论电影在线| 99久久精品99国产精品| 国产高清在线精品| 久久精品久久99精品久久| 亚洲国产欧美一区二区三区丁香婷| 精品欧美乱码久久久久久1区2区| 欧美在线观看视频一区二区三区| 国产aⅴ精品一区二区三区色成熟| 日本不卡一区二区三区高清视频| 亚洲人成小说网站色在线 | 99re热这里只有精品视频| 国产一区二区三区观看| 激情久久久久久久久久久久久久久久 | www..com久久爱| 99久久伊人久久99| a4yy欧美一区二区三区| 99视频一区二区三区| 色综合久久天天| 欧洲视频一区二区| 欧美久久高跟鞋激| 欧美一区二区三区婷婷月色| 欧美日本精品一区二区三区| 精品视频在线免费| 91精品一区二区三区在线观看| 91麻豆精品国产无毒不卡在线观看| 91精品国产综合久久小美女| 精品国产青草久久久久福利| 日韩欧美国产综合一区 | 国产精品久久看| 亚洲免费观看高清完整版在线| 亚洲午夜激情av| 久久精品国产77777蜜臀| 成人黄色777网| 欧美日韩一级视频| 国产日韩欧美a| 石原莉奈一区二区三区在线观看| 国产精品1区2区| 91久久精品一区二区三| 精品国产免费一区二区三区香蕉| 国产精品久久三| 另类小说色综合网站| 色视频成人在线观看免| 26uuu久久综合| 美女爽到高潮91| 欧美性感一类影片在线播放| 亚洲国产精品成人综合色在线婷婷| 一区二区三区中文免费| 成人高清视频在线| 久久久久高清精品| 久草精品在线观看| 欧美挠脚心视频网站| 亚洲品质自拍视频网站| 免费成人在线观看视频| 午夜精品国产更新| www.欧美.com| 国产精品美日韩| 国产精品资源网站| 久久久久久久久99精品| 久久成人羞羞网站| 日韩精品一区二区三区在线| 天天操天天干天天综合网| 精品视频在线免费观看| 亚洲一区二区三区视频在线播放| 一本一道久久a久久精品| 亚洲天堂精品视频| 欧洲精品在线观看| 图片区日韩欧美亚洲| 精品视频1区2区3区| 秋霞av亚洲一区二区三| 日韩欧美123| 不卡的av在线播放| 亚洲人妖av一区二区|