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

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

?? hid-tmff.c

?? Linux Kernel 2.6.9 for OMAP1710
?? C
字號:
/* * 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
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
一区二区三区四区激情| 欧美一级精品大片| 日韩黄色片在线观看| 日韩欧美国产精品| 91久久人澡人人添人人爽欧美 | 日韩欧美一区二区在线视频| 成人中文字幕在线| 日韩一区在线看| 久久综合九色综合欧美亚洲| 欧美日韩国产一级| 97久久超碰国产精品电影| 在线观看日韩毛片| 国产jizzjizz一区二区| 久久精品噜噜噜成人av农村| 亚洲一区二区三区四区在线| 国产精品国产三级国产普通话蜜臀 | 亚洲大型综合色站| 亚洲欧洲在线观看av| 久久久一区二区| 日韩欧美国产午夜精品| 欧美丰满高潮xxxx喷水动漫| 日韩国产精品久久久久久亚洲| 男女激情视频一区| 久久久久久久久久久久电影 | 亚洲精品一区二区三区精华液| 色婷婷久久久综合中文字幕 | 亚洲成av人在线观看| 国产精品美女久久久久久久网站| 久久免费午夜影院| 久久综合久色欧美综合狠狠| 日韩午夜精品视频| 91麻豆精品国产91久久久久久久久 | 一区二区三区四区在线播放| 国产精品国产精品国产专区不蜜| 国产偷v国产偷v亚洲高清| 欧美一区二区啪啪| 欧美一区二区播放| 欧美一区二区三区男人的天堂| 制服.丝袜.亚洲.另类.中文| 欧美剧情片在线观看| 在线成人av网站| 91精品国产综合久久国产大片| 欧美日本乱大交xxxxx| 91精品国产综合久久福利软件| 欧美一区二区视频网站| 亚洲女爱视频在线| 亚洲人成在线播放网站岛国| 一区二区三区产品免费精品久久75| 悠悠色在线精品| 爽爽淫人综合网网站| 麻豆91免费看| 国产原创一区二区| 成人av网站在线观看| 91在线视频在线| 在线视频国内自拍亚洲视频| 欧美精品日韩精品| 欧美精品一区二区三区在线| wwwwww.欧美系列| 国产精品欧美经典| 一区二区三区欧美视频| 日本视频一区二区| 国产经典欧美精品| 欧洲一区二区av| 日韩视频在线永久播放| 国产视频一区不卡| 亚洲精品视频自拍| 日本aⅴ免费视频一区二区三区| 久久99精品久久久久久动态图| 国产91丝袜在线播放0| 99久久精品国产导航| 欧美欧美欧美欧美首页| 337p粉嫩大胆噜噜噜噜噜91av | 精品久久久久久久久久久久久久久| 亚洲精品在线电影| 国产精品福利在线播放| 亚洲成av人片观看| 国内国产精品久久| 91成人免费在线视频| 精品国产污网站| 国产精品久久久久精k8| 午夜精品久久久久久久蜜桃app| 国产精品影视在线| 欧美三区免费完整视频在线观看| 欧美videos中文字幕| 亚洲蜜臀av乱码久久精品蜜桃| 久久国产精品第一页| 99国产精品久久久久| 欧美sm极限捆绑bd| 夜夜精品视频一区二区| 国产一区二区看久久| 欧美色倩网站大全免费| 国产三级三级三级精品8ⅰ区| 亚洲大片精品永久免费| 国产91精品露脸国语对白| 欧美另类久久久品| 国产精品福利一区二区三区| 久久99日本精品| 欧美性猛交xxxx黑人交| 国产日产欧美一区二区视频| 欧美aaa在线| 国产精品久99| 寂寞少妇一区二区三区| 在线观看一区二区精品视频| 欧美高清在线一区| 久久99九九99精品| 欧美日韩一区二区电影| 日韩美女视频19| 国产99一区视频免费 | 国产丝袜美腿一区二区三区| 青青国产91久久久久久| 欧美日韩一区二区三区在线| 欧美经典一区二区三区| 亚洲综合自拍偷拍| 国产成人av在线影院| 欧美老年两性高潮| 亚洲人成精品久久久久| 国产成人在线视频网站| 精品国产乱码久久久久久1区2区| 午夜在线电影亚洲一区| 在线一区二区三区四区| 国产精品丝袜在线| 国产精品99久久久久久有的能看| 日韩欧美色电影| 热久久国产精品| 91精品国产免费| 秋霞电影网一区二区| 欧美高清视频www夜色资源网| 亚洲综合色自拍一区| 日本丶国产丶欧美色综合| 中文字幕一区二区三区四区不卡 | 亚洲激情在线激情| 97aⅴ精品视频一二三区| 国产精品青草久久| 成人午夜激情在线| 国产精品国产a| 97se亚洲国产综合自在线不卡| 成人免费小视频| 99久久夜色精品国产网站| 国产精品日韩成人| 91视频国产观看| 亚洲一区二区三区国产| 在线成人av网站| 蜜桃av一区二区在线观看| 91精品国产色综合久久不卡电影 | 日本黄色一区二区| 一卡二卡三卡日韩欧美| 欧洲精品一区二区| 亚洲成人av免费| 欧美一级高清片在线观看| 久久av中文字幕片| 中文字幕精品在线不卡| 成人蜜臀av电影| 久久久三级国产网站| 天天综合日日夜夜精品| 91精品国产美女浴室洗澡无遮挡| 久久99精品国产麻豆婷婷 | 亚洲欧洲99久久| 欧洲激情一区二区| 91偷拍与自偷拍精品| 亚洲激情一二三区| 欧美一区二区女人| 国产成人精品aa毛片| 最新热久久免费视频| 在线观看日产精品| 免费xxxx性欧美18vr| 欧美国产精品v| 在线观看一区二区视频| 男人操女人的视频在线观看欧美| 2021久久国产精品不只是精品| 不卡一二三区首页| 五月天亚洲婷婷| 久久久精品综合| 欧美性videosxxxxx| 欧美激情一区二区三区全黄| 成人听书哪个软件好| 亚洲精品欧美激情| 欧美一区二区三区免费大片| 国产不卡视频在线播放| 亚洲成av人片在线| 国产欧美精品在线观看| 欧美亚洲国产一卡| 国产乱码精品1区2区3区| 亚洲欧洲精品一区二区三区| 欧美精选在线播放| 国产a久久麻豆| 欧美a一区二区| 亚洲欧美日本韩国| 久久蜜桃一区二区| 69av一区二区三区| 国产白丝网站精品污在线入口| 亚洲va国产天堂va久久en| 欧美激情一区二区在线| 91精品国产综合久久久久久久| 成人精品视频网站| 麻豆精品久久久| 亚洲一级二级在线| 国产精品女同互慰在线看| 欧美tickling网站挠脚心| 欧洲日韩一区二区三区| 国产成人午夜99999|