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

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

?? hid-tmff.c

?? 底層驅(qū)動開發(fā)
?? 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);}

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
国产成人综合亚洲91猫咪| 欧美日韩成人综合在线一区二区| 色www精品视频在线观看| 欧美一区日韩一区| 亚洲女同一区二区| 国产精品亚洲人在线观看| 在线不卡一区二区| 亚洲色图视频网| 国产91丝袜在线播放九色| 制服丝袜成人动漫| 亚洲一区视频在线| 9l国产精品久久久久麻豆| 欧美一级二级三级蜜桃| 亚洲女同一区二区| 不卡视频在线观看| 国产欧美一区二区精品婷婷| 麻豆精品视频在线观看| 欧美综合亚洲图片综合区| 一区二区中文字幕在线| 国产精品一二三区在线| 精品理论电影在线| 六月婷婷色综合| 欧美一级专区免费大片| 亚洲主播在线播放| 91麻豆视频网站| 中文字幕制服丝袜一区二区三区| 国产馆精品极品| 国产亚洲精品福利| 国产成人综合网站| 久久久精品欧美丰满| 紧缚捆绑精品一区二区| 日韩美一区二区三区| 日本伊人精品一区二区三区观看方式| 欧美影视一区在线| 亚洲成人自拍网| 欧美午夜理伦三级在线观看| 午夜成人免费视频| 国产成人8x视频一区二区| 18涩涩午夜精品.www| 美女脱光内衣内裤视频久久网站 | 在线观看日韩毛片| 国产一区欧美日韩| 亚洲制服欧美中文字幕中文字幕| 欧美一区二区久久| 99精品久久免费看蜜臀剧情介绍| 欧美a级一区二区| 亚洲欧美另类久久久精品2019| 欧美一卡在线观看| 欧美性受极品xxxx喷水| 国产伦理精品不卡| 日韩福利视频导航| 亚洲乱码日产精品bd| 久久久美女毛片| 91精品福利在线一区二区三区| 91麻豆免费观看| 成人午夜精品一区二区三区| 麻豆高清免费国产一区| 亚洲综合一区在线| 亚洲免费观看高清完整版在线 | 中文字幕一区在线观看| 欧美精品一区二区三区很污很色的 | 欧美日韩一级二级三级| 成人性生交大合| 国产乱子轮精品视频| 亚洲成人精品影院| 亚洲精品视频一区| 亚洲欧美综合色| 欧美国产成人在线| 久久久91精品国产一区二区精品 | 一本在线高清不卡dvd| 国产91丝袜在线播放| 韩国av一区二区三区| 精品成人一区二区| 色999日韩国产欧美一区二区| 国产露脸91国语对白| 国模少妇一区二区三区| 极品少妇一区二区| 九色综合国产一区二区三区| 人禽交欧美网站| 美国三级日本三级久久99| 日本欧美一区二区在线观看| 婷婷国产在线综合| 欧美aaa在线| 黄网站免费久久| 国产老女人精品毛片久久| 国产精品一区二区你懂的| 国产一区不卡精品| 粉嫩绯色av一区二区在线观看| 国产成人在线免费观看| 成人在线视频一区二区| 成人午夜碰碰视频| 91热门视频在线观看| 在线国产亚洲欧美| 欧美精品一二三四| 欧美一卡二卡在线| 国产偷v国产偷v亚洲高清| 国产精品伦理在线| 亚洲一区中文日韩| 日本成人在线电影网| 国产一区二区在线视频| 成人h动漫精品| 在线免费观看日本欧美| 7777精品伊人久久久大香线蕉完整版| 欧美一二三区在线| 久久久不卡网国产精品一区| 中文字幕一区二区三区蜜月| 亚洲午夜私人影院| 精品一区二区三区不卡| 成人av网站在线观看| 欧洲一区在线观看| 亚洲精品在线电影| 亚洲欧美日韩在线| 麻豆91免费观看| 波多野结衣中文字幕一区| 欧美性视频一区二区三区| 日韩欧美一区二区在线视频| 国产精品网曝门| 午夜不卡av在线| 成人福利视频网站| 欧美一二三区精品| 亚洲美女淫视频| 韩国女主播一区二区三区| 色噜噜狠狠成人中文综合| 精品精品国产高清a毛片牛牛| 综合久久久久久| 久久99精品视频| 欧美日韩亚洲国产综合| 久久精品视频在线免费观看 | 91精品久久久久久久99蜜桃| 国产欧美日韩不卡免费| 日韩影院精彩在线| 色综合久久综合网| 国产午夜精品在线观看| 日日夜夜精品视频免费| 91一区二区在线| 久久久亚洲国产美女国产盗摄 | 日韩电影免费在线| 91亚洲精品一区二区乱码| 精品少妇一区二区三区| 亚洲第一二三四区| 91麻豆国产精品久久| www国产精品av| 日日夜夜免费精品| 91国偷自产一区二区三区观看| 国产亚洲短视频| 麻豆成人免费电影| 欧美一区二区视频在线观看2022 | 精品免费日韩av| 天天操天天干天天综合网| 一本一本久久a久久精品综合麻豆| 久久久精品国产免大香伊| 黄页网站大全一区二区| 91精品国产91热久久久做人人| 亚洲一区在线观看网站| 在线观看日韩国产| 亚洲精品一二三四区| 99re8在线精品视频免费播放| 久久久精品免费网站| 激情综合网av| 久久久精品综合| 国产成人午夜视频| 国产午夜亚洲精品羞羞网站| 国产乱码精品一区二区三区五月婷| 欧美videofree性高清杂交| 麻豆精品视频在线| 精品国产精品网麻豆系列| 久久精品理论片| 欧美一区欧美二区| 免费一级片91| 欧美日韩aaa| 久久99国内精品| 日韩视频不卡中文| 视频在线在亚洲| 精品国产一区二区三区四区四| 日韩精品成人一区二区三区| 一本一本久久a久久精品综合麻豆| 欧洲一区在线电影| 三级久久三级久久久| 欧美日韩精品欧美日韩精品一综合| 亚洲免费在线视频| 欧美三级视频在线| 一区二区三区资源| 色偷偷一区二区三区| 国产欧美一区视频| 91在线视频18| 亚洲人妖av一区二区| www.66久久| 婷婷久久综合九色国产成人| 欧美日韩你懂的| 午夜影院久久久| 欧美日韩亚洲综合在线 欧美亚洲特黄一级| 一区二区三区自拍| 欧美日韩国产综合一区二区| 亚洲精品国产精品乱码不99| 91亚洲精品乱码久久久久久蜜桃| 午夜精品一区二区三区三上悠亚| 日本精品一区二区三区四区的功能| 亚洲国产经典视频| 国产福利一区二区三区在线视频| 国产欧美日韩综合精品一区二区|