ams-input.c 3.3 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
/*
 * Apple Motion Sensor driver (joystick emulation)
 *
 * Copyright (C) 2005 Stelian Pop (stelian@popies.net)
 * Copyright (C) 2006 Michael Hanselmann (linux-kernel@hansmi.ch)
 *
 * 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.
 */

#include <linux/module.h>

#include <linux/types.h>
#include <linux/errno.h>
#include <linux/init.h>
#include <linux/delay.h>

#include "ams.h"

static unsigned int joystick;
23
module_param(joystick, bool, S_IRUGO);
24 25 26
MODULE_PARM_DESC(joystick, "Enable the input class device on module load");

static unsigned int invert;
27
module_param(invert, bool, S_IWUSR | S_IRUGO);
28 29
MODULE_PARM_DESC(invert, "Invert input data on X and Y axis");

30
static void ams_idev_poll(struct input_polled_dev *dev)
31
{
32
	struct input_dev *idev = dev->input;
33 34
	s8 x, y, z;

35
	mutex_lock(&ams_info.lock);
36

37
	ams_sensors(&x, &y, &z);
38

39 40 41
	x -= ams_info.xcalib;
	y -= ams_info.ycalib;
	z -= ams_info.zcalib;
42

43 44 45
	input_report_abs(idev, ABS_X, invert ? -x : x);
	input_report_abs(idev, ABS_Y, invert ? -y : y);
	input_report_abs(idev, ABS_Z, z);
46

47
	input_sync(idev);
48

49
	mutex_unlock(&ams_info.lock);
50 51 52 53 54
}

/* Call with ams_info.lock held! */
static void ams_input_enable(void)
{
55
	struct input_dev *input;
56 57 58 59 60 61 62 63 64 65
	s8 x, y, z;

	if (ams_info.idev)
		return;

	ams_sensors(&x, &y, &z);
	ams_info.xcalib = x;
	ams_info.ycalib = y;
	ams_info.zcalib = z;

66
	ams_info.idev = input_allocate_polled_device();
67 68 69
	if (!ams_info.idev)
		return;

70 71 72 73 74 75 76 77
	ams_info.idev->poll = ams_idev_poll;
	ams_info.idev->poll_interval = 25;

	input = ams_info.idev->input;
	input->name = "Apple Motion Sensor";
	input->id.bustype = ams_info.bustype;
	input->id.vendor = 0;
	input->dev.parent = &ams_info.of_dev->dev;
78

79 80 81
	input_set_abs_params(input, ABS_X, -50, 50, 3, 0);
	input_set_abs_params(input, ABS_Y, -50, 50, 3, 0);
	input_set_abs_params(input, ABS_Z, -50, 50, 3, 0);
82

83 84 85
	set_bit(EV_ABS, input->evbit);
	set_bit(EV_KEY, input->evbit);
	set_bit(BTN_TOUCH, input->keybit);
86

87 88
	if (input_register_polled_device(ams_info.idev)) {
		input_free_polled_device(ams_info.idev);
89 90 91 92 93 94 95 96 97
		ams_info.idev = NULL;
		return;
	}
}

/* Call with ams_info.lock held! */
static void ams_input_disable(void)
{
	if (ams_info.idev) {
98 99
		input_unregister_polled_device(ams_info.idev);
		input_free_polled_device(ams_info.idev);
100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
		ams_info.idev = NULL;
	}
}

static ssize_t ams_input_show_joystick(struct device *dev,
	struct device_attribute *attr, char *buf)
{
	return sprintf(buf, "%d\n", joystick);
}

static ssize_t ams_input_store_joystick(struct device *dev,
	struct device_attribute *attr, const char *buf, size_t count)
{
	if (sscanf(buf, "%d\n", &joystick) != 1)
		return -EINVAL;

	mutex_lock(&ams_info.lock);

	if (joystick)
		ams_input_enable();
	else
		ams_input_disable();

	mutex_unlock(&ams_info.lock);

	return count;
}

static DEVICE_ATTR(joystick, S_IRUGO | S_IWUSR,
	ams_input_show_joystick, ams_input_store_joystick);

/* Call with ams_info.lock held! */
int ams_input_init(void)
{
	int result;

	result = device_create_file(&ams_info.of_dev->dev, &dev_attr_joystick);

	if (!result && joystick)
		ams_input_enable();
	return result;
}

/* Call with ams_info.lock held! */
A
Al Viro 已提交
144
void ams_input_exit(void)
145 146 147 148
{
	ams_input_disable();
	device_remove_file(&ams_info.of_dev->dev, &dev_attr_joystick);
}