lis3lv02d.c 20.5 KB
Newer Older
P
Pavel Machek 已提交
1 2 3 4 5
/*
 *  lis3lv02d.c - ST LIS3LV02DL accelerometer driver
 *
 *  Copyright (C) 2007-2008 Yan Burman
 *  Copyright (C) 2008 Eric Piel
6
 *  Copyright (C) 2008-2009 Pavel Machek
P
Pavel Machek 已提交
7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
 *
 *  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/kernel.h>
#include <linux/init.h>
#include <linux/dmi.h>
#include <linux/module.h>
#include <linux/types.h>
#include <linux/platform_device.h>
#include <linux/interrupt.h>
E
Eric Piel 已提交
30
#include <linux/input-polldev.h>
P
Pavel Machek 已提交
31 32 33 34 35
#include <linux/delay.h>
#include <linux/wait.h>
#include <linux/poll.h>
#include <linux/freezer.h>
#include <linux/uaccess.h>
36
#include <linux/miscdevice.h>
P
Pavel Machek 已提交
37 38 39 40 41 42 43
#include <asm/atomic.h>
#include "lis3lv02d.h"

#define DRIVER_NAME     "lis3lv02d"

/* joystick device poll interval in milliseconds */
#define MDPS_POLL_INTERVAL 50
S
Samu Onkalo 已提交
44 45
#define MDPS_POLL_MIN	   0
#define MDPS_POLL_MAX	   2000
P
Pavel Machek 已提交
46 47
/*
 * The sensor can also generate interrupts (DRDY) but it's pretty pointless
48
 * because they are generated even if the data do not change. So it's better
P
Pavel Machek 已提交
49 50 51 52 53 54
 * to keep the interrupt for the free-fall event. The values are updated at
 * 40Hz (at the lowest frequency), but as it can be pretty time consuming on
 * some low processor, we poll the sensor only at 20Hz... enough for the
 * joystick.
 */

S
Samu Onkalo 已提交
55 56 57
#define LIS3_PWRON_DELAY_WAI_12B	(5000)
#define LIS3_PWRON_DELAY_WAI_8B		(3000)

S
Samu Onkalo 已提交
58 59 60 61 62 63 64 65 66 67 68 69 70 71
/*
 * LIS3LV02D spec says 1024 LSBs corresponds 1 G -> 1LSB is 1000/1024 mG
 * LIS302D spec says: 18 mG / digit
 * LIS3_ACCURACY is used to increase accuracy of the intermediate
 * calculation results.
 */
#define LIS3_ACCURACY			1024
/* Sensitivity values for -2G +2G scale */
#define LIS3_SENSITIVITY_12B		((LIS3_ACCURACY * 1000) / 1024)
#define LIS3_SENSITIVITY_8B		(18 * LIS3_ACCURACY)

#define LIS3_DEFAULT_FUZZ		3
#define LIS3_DEFAULT_FLAT		3

72
struct lis3lv02d lis3_dev = {
73
	.misc_wait   = __WAIT_QUEUE_HEAD_INITIALIZER(lis3_dev.misc_wait),
74 75
};

76
EXPORT_SYMBOL_GPL(lis3_dev);
P
Pavel Machek 已提交
77

78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101
/* just like param_set_int() but does sanity-check so that it won't point
 * over the axis array size
 */
static int param_set_axis(const char *val, const struct kernel_param *kp)
{
	int ret = param_set_int(val, kp);
	if (!ret) {
		int val = *(int *)kp->arg;
		if (val < 0)
			val = -val;
		if (!val || val > 3)
			return -EINVAL;
	}
	return ret;
}

static struct kernel_param_ops param_ops_axis = {
	.set = param_set_axis,
	.get = param_get_int,
};

module_param_array_named(axes, lis3_dev.ac.as_array, axis, NULL, 0644);
MODULE_PARM_DESC(axes, "Axis-mapping for x,y,z directions");

102 103 104 105 106 107 108 109 110
static s16 lis3lv02d_read_8(struct lis3lv02d *lis3, int reg)
{
	s8 lo;
	if (lis3->read(lis3, reg, &lo) < 0)
		return 0;

	return lo;
}

111
static s16 lis3lv02d_read_12(struct lis3lv02d *lis3, int reg)
112 113 114
{
	u8 lo, hi;

115 116
	lis3->read(lis3, reg - 1, &lo);
	lis3->read(lis3, reg, &hi);
117 118 119 120
	/* In "12 bit right justified" mode, bit 6, bit 7, bit 8 = bit 5 */
	return (s16)((hi << 8) | lo);
}

P
Pavel Machek 已提交
121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137
/**
 * lis3lv02d_get_axis - For the given axis, give the value converted
 * @axis:      1,2,3 - can also be negative
 * @hw_values: raw values returned by the hardware
 *
 * Returns the converted value.
 */
static inline int lis3lv02d_get_axis(s8 axis, int hw_values[3])
{
	if (axis > 0)
		return hw_values[axis - 1];
	else
		return -hw_values[-axis - 1];
}

/**
 * lis3lv02d_get_xyz - Get X, Y and Z axis values from the accelerometer
138 139 140 141
 * @lis3: pointer to the device struct
 * @x:    where to store the X axis value
 * @y:    where to store the Y axis value
 * @z:    where to store the Z axis value
P
Pavel Machek 已提交
142 143 144
 *
 * Note that 40Hz input device can eat up about 10% CPU at 800MHZ
 */
145
static void lis3lv02d_get_xyz(struct lis3lv02d *lis3, int *x, int *y, int *z)
P
Pavel Machek 已提交
146 147
{
	int position[3];
S
Samu Onkalo 已提交
148
	int i;
P
Pavel Machek 已提交
149

150 151 152
	position[0] = lis3->read_data(lis3, OUTX);
	position[1] = lis3->read_data(lis3, OUTY);
	position[2] = lis3->read_data(lis3, OUTZ);
P
Pavel Machek 已提交
153

S
Samu Onkalo 已提交
154 155 156
	for (i = 0; i < 3; i++)
		position[i] = (position[i] * lis3->scale) / LIS3_ACCURACY;

157 158 159
	*x = lis3lv02d_get_axis(lis3->ac.x, position);
	*y = lis3lv02d_get_axis(lis3->ac.y, position);
	*z = lis3lv02d_get_axis(lis3->ac.z, position);
P
Pavel Machek 已提交
160 161
}

S
Samu Onkalo 已提交
162 163 164
/* conversion btw sampling rate and the register values */
static int lis3_12_rates[4] = {40, 160, 640, 2560};
static int lis3_8_rates[2] = {100, 400};
165
static int lis3_3dc_rates[16] = {0, 1, 10, 25, 50, 100, 200, 400, 1600, 5000};
S
Samu Onkalo 已提交
166

167
/* ODR is Output Data Rate */
S
Samu Onkalo 已提交
168 169 170
static int lis3lv02d_get_odr(void)
{
	u8 ctrl;
171
	int shift;
S
Samu Onkalo 已提交
172 173

	lis3_dev.read(&lis3_dev, CTRL_REG1, &ctrl);
174 175 176 177
	ctrl &= lis3_dev.odr_mask;
	shift = ffs(lis3_dev.odr_mask) - 1;
	return lis3_dev.odrs[(ctrl >> shift)];
}
S
Samu Onkalo 已提交
178

179 180 181 182 183
static int lis3lv02d_set_odr(int rate)
{
	u8 ctrl;
	int i, len, shift;

184 185 186
	if (!rate)
		return -EINVAL;

187 188 189 190 191 192 193 194 195 196 197 198
	lis3_dev.read(&lis3_dev, CTRL_REG1, &ctrl);
	ctrl &= ~lis3_dev.odr_mask;
	len = 1 << hweight_long(lis3_dev.odr_mask); /* # of possible values */
	shift = ffs(lis3_dev.odr_mask) - 1;

	for (i = 0; i < len; i++)
		if (lis3_dev.odrs[i] == rate) {
			lis3_dev.write(&lis3_dev, CTRL_REG1,
					ctrl | (i << shift));
			return 0;
		}
	return -EINVAL;
S
Samu Onkalo 已提交
199 200
}

S
Samu Onkalo 已提交
201 202
static int lis3lv02d_selftest(struct lis3lv02d *lis3, s16 results[3])
{
203
	u8 ctlreg, reg;
S
Samu Onkalo 已提交
204 205 206 207 208
	s16 x, y, z;
	u8 selftest;
	int ret;

	mutex_lock(&lis3->mutex);
209 210 211 212 213 214 215 216 217 218
	if (lis3_dev.whoami == WAI_3DC) {
		ctlreg = CTRL_REG4;
		selftest = CTRL4_ST0;
	} else {
		ctlreg = CTRL_REG1;
		if (lis3_dev.whoami == WAI_12B)
			selftest = CTRL1_ST;
		else
			selftest = CTRL1_STP;
	}
S
Samu Onkalo 已提交
219

220 221
	lis3->read(lis3, ctlreg, &reg);
	lis3->write(lis3, ctlreg, (reg | selftest));
S
Samu Onkalo 已提交
222 223 224 225 226 227 228 229
	msleep(lis3->pwron_delay / lis3lv02d_get_odr());

	/* Read directly to avoid axis remap */
	x = lis3->read_data(lis3, OUTX);
	y = lis3->read_data(lis3, OUTY);
	z = lis3->read_data(lis3, OUTZ);

	/* back to normal settings */
230
	lis3->write(lis3, ctlreg, reg);
S
Samu Onkalo 已提交
231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255
	msleep(lis3->pwron_delay / lis3lv02d_get_odr());

	results[0] = x - lis3->read_data(lis3, OUTX);
	results[1] = y - lis3->read_data(lis3, OUTY);
	results[2] = z - lis3->read_data(lis3, OUTZ);

	ret = 0;
	if (lis3->pdata) {
		int i;
		for (i = 0; i < 3; i++) {
			/* Check against selftest acceptance limits */
			if ((results[i] < lis3->pdata->st_min_limits[i]) ||
			    (results[i] > lis3->pdata->st_max_limits[i])) {
				ret = -EIO;
				goto fail;
			}
		}
	}

	/* test passed */
fail:
	mutex_unlock(&lis3->mutex);
	return ret;
}

256
void lis3lv02d_poweroff(struct lis3lv02d *lis3)
P
Pavel Machek 已提交
257
{
258 259
	/* disable X,Y,Z axis and power down */
	lis3->write(lis3, CTRL_REG1, 0x00);
P
Pavel Machek 已提交
260
}
261
EXPORT_SYMBOL_GPL(lis3lv02d_poweroff);
P
Pavel Machek 已提交
262

263
void lis3lv02d_poweron(struct lis3lv02d *lis3)
P
Pavel Machek 已提交
264
{
265
	u8 reg;
P
Pavel Machek 已提交
266

267
	lis3->init(lis3);
P
Pavel Machek 已提交
268

S
Samu Onkalo 已提交
269 270 271
	/* LIS3 power on delay is quite long */
	msleep(lis3->pwron_delay / lis3lv02d_get_odr());

272 273
	/*
	 * Common configuration
É
Éric Piel 已提交
274 275
	 * BDU: (12 bits sensors only) LSB and MSB values are not updated until
	 *      both have been read. So the value read will always be correct.
276
	 */
É
Éric Piel 已提交
277 278 279 280 281
	if (lis3->whoami ==  WAI_12B) {
		lis3->read(lis3, CTRL_REG2, &reg);
		reg |= CTRL2_BDU;
		lis3->write(lis3, CTRL_REG2, reg);
	}
P
Pavel Machek 已提交
282
}
283 284
EXPORT_SYMBOL_GPL(lis3lv02d_poweron);

P
Pavel Machek 已提交
285

286 287 288 289 290 291 292 293 294 295 296 297 298
static void lis3lv02d_joystick_poll(struct input_polled_dev *pidev)
{
	int x, y, z;

	mutex_lock(&lis3_dev.mutex);
	lis3lv02d_get_xyz(&lis3_dev, &x, &y, &z);
	input_report_abs(pidev->input, ABS_X, x);
	input_report_abs(pidev->input, ABS_Y, y);
	input_report_abs(pidev->input, ABS_Z, z);
	input_sync(pidev->input);
	mutex_unlock(&lis3_dev.mutex);
}

299 300
static irqreturn_t lis302dl_interrupt(int irq, void *dummy)
{
301 302 303
	if (!test_bit(0, &lis3_dev.misc_opened))
		goto out;

304 305 306 307 308
	/*
	 * Be careful: on some HP laptops the bios force DD when on battery and
	 * the lid is closed. This leads to interrupts as soon as a little move
	 * is done.
	 */
309
	atomic_inc(&lis3_dev.count);
310

311 312
	wake_up_interruptible(&lis3_dev.misc_wait);
	kill_fasync(&lis3_dev.async_queue, SIGIO, POLL_IN);
313
out:
314
	if (lis3_dev.pdata && lis3_dev.whoami == WAI_8B && lis3_dev.idev &&
315 316
	    lis3_dev.idev->input->users)
		return IRQ_WAKE_THREAD;
317 318 319
	return IRQ_HANDLED;
}

320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361
static void lis302dl_interrupt_handle_click(struct lis3lv02d *lis3)
{
	struct input_dev *dev = lis3->idev->input;
	u8 click_src;

	mutex_lock(&lis3->mutex);
	lis3->read(lis3, CLICK_SRC, &click_src);

	if (click_src & CLICK_SINGLE_X) {
		input_report_key(dev, lis3->mapped_btns[0], 1);
		input_report_key(dev, lis3->mapped_btns[0], 0);
	}

	if (click_src & CLICK_SINGLE_Y) {
		input_report_key(dev, lis3->mapped_btns[1], 1);
		input_report_key(dev, lis3->mapped_btns[1], 0);
	}

	if (click_src & CLICK_SINGLE_Z) {
		input_report_key(dev, lis3->mapped_btns[2], 1);
		input_report_key(dev, lis3->mapped_btns[2], 0);
	}
	input_sync(dev);
	mutex_unlock(&lis3->mutex);
}

static void lis302dl_interrupt_handle_ff_wu(struct lis3lv02d *lis3)
{
	u8 wu1_src;
	u8 wu2_src;

	lis3->read(lis3, FF_WU_SRC_1, &wu1_src);
	lis3->read(lis3, FF_WU_SRC_2, &wu2_src);

	wu1_src = wu1_src & FF_WU_SRC_IA ? wu1_src : 0;
	wu2_src = wu2_src & FF_WU_SRC_IA ? wu2_src : 0;

	/* joystick poll is internally protected by the lis3->mutex. */
	if (wu1_src || wu2_src)
		lis3lv02d_joystick_poll(lis3_dev.idev);
}

362
static irqreturn_t lis302dl_interrupt_thread1_8b(int irq, void *data)
363
{
364 365 366 367 368 369 370 371

	struct lis3lv02d *lis3 = data;

	if ((lis3->pdata->irq_cfg & LIS3_IRQ1_MASK) == LIS3_IRQ1_CLICK)
		lis302dl_interrupt_handle_click(lis3);
	else
		lis302dl_interrupt_handle_ff_wu(lis3);

372 373
	return IRQ_HANDLED;
}
374

375 376
static irqreturn_t lis302dl_interrupt_thread2_8b(int irq, void *data)
{
377 378 379 380 381 382 383 384

	struct lis3lv02d *lis3 = data;

	if ((lis3->pdata->irq_cfg & LIS3_IRQ2_MASK) == LIS3_IRQ2_CLICK)
		lis302dl_interrupt_handle_click(lis3);
	else
		lis302dl_interrupt_handle_ff_wu(lis3);

385 386 387 388 389
	return IRQ_HANDLED;
}

static int lis3lv02d_misc_open(struct inode *inode, struct file *file)
{
390
	if (test_and_set_bit(0, &lis3_dev.misc_opened))
391 392
		return -EBUSY; /* already open */

393
	atomic_set(&lis3_dev.count, 0);
394 395 396 397 398
	return 0;
}

static int lis3lv02d_misc_release(struct inode *inode, struct file *file)
{
399 400
	fasync_helper(-1, file, 0, &lis3_dev.async_queue);
	clear_bit(0, &lis3_dev.misc_opened); /* release the device */
401 402 403 404 405 406 407 408 409 410 411 412 413 414
	return 0;
}

static ssize_t lis3lv02d_misc_read(struct file *file, char __user *buf,
				size_t count, loff_t *pos)
{
	DECLARE_WAITQUEUE(wait, current);
	u32 data;
	unsigned char byte_data;
	ssize_t retval = 1;

	if (count < 1)
		return -EINVAL;

415
	add_wait_queue(&lis3_dev.misc_wait, &wait);
416 417
	while (true) {
		set_current_state(TASK_INTERRUPTIBLE);
418
		data = atomic_xchg(&lis3_dev.count, 0);
419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447
		if (data)
			break;

		if (file->f_flags & O_NONBLOCK) {
			retval = -EAGAIN;
			goto out;
		}

		if (signal_pending(current)) {
			retval = -ERESTARTSYS;
			goto out;
		}

		schedule();
	}

	if (data < 255)
		byte_data = data;
	else
		byte_data = 255;

	/* make sure we are not going into copy_to_user() with
	 * TASK_INTERRUPTIBLE state */
	set_current_state(TASK_RUNNING);
	if (copy_to_user(buf, &byte_data, sizeof(byte_data)))
		retval = -EFAULT;

out:
	__set_current_state(TASK_RUNNING);
448
	remove_wait_queue(&lis3_dev.misc_wait, &wait);
449 450 451 452 453 454

	return retval;
}

static unsigned int lis3lv02d_misc_poll(struct file *file, poll_table *wait)
{
455 456
	poll_wait(file, &lis3_dev.misc_wait, wait);
	if (atomic_read(&lis3_dev.count))
457 458 459 460 461 462
		return POLLIN | POLLRDNORM;
	return 0;
}

static int lis3lv02d_misc_fasync(int fd, struct file *file, int on)
{
463
	return fasync_helper(fd, file, on, &lis3_dev.async_queue);
464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481
}

static const struct file_operations lis3lv02d_misc_fops = {
	.owner   = THIS_MODULE,
	.llseek  = no_llseek,
	.read    = lis3lv02d_misc_read,
	.open    = lis3lv02d_misc_open,
	.release = lis3lv02d_misc_release,
	.poll    = lis3lv02d_misc_poll,
	.fasync  = lis3lv02d_misc_fasync,
};

static struct miscdevice lis3lv02d_misc_device = {
	.minor   = MISC_DYNAMIC_MINOR,
	.name    = "freefall",
	.fops    = &lis3lv02d_misc_fops,
};

482
int lis3lv02d_joystick_enable(void)
P
Pavel Machek 已提交
483
{
E
Eric Piel 已提交
484
	struct input_dev *input_dev;
P
Pavel Machek 已提交
485
	int err;
S
Samu Onkalo 已提交
486
	int max_val, fuzz, flat;
487
	int btns[] = {BTN_X, BTN_Y, BTN_Z};
P
Pavel Machek 已提交
488

489
	if (lis3_dev.idev)
P
Pavel Machek 已提交
490 491
		return -EINVAL;

E
Eric Piel 已提交
492
	lis3_dev.idev = input_allocate_polled_device();
493
	if (!lis3_dev.idev)
P
Pavel Machek 已提交
494 495
		return -ENOMEM;

E
Eric Piel 已提交
496 497
	lis3_dev.idev->poll = lis3lv02d_joystick_poll;
	lis3_dev.idev->poll_interval = MDPS_POLL_INTERVAL;
S
Samu Onkalo 已提交
498 499
	lis3_dev.idev->poll_interval_min = MDPS_POLL_MIN;
	lis3_dev.idev->poll_interval_max = MDPS_POLL_MAX;
E
Eric Piel 已提交
500 501 502 503 504 505 506
	input_dev = lis3_dev.idev->input;

	input_dev->name       = "ST LIS3LV02DL Accelerometer";
	input_dev->phys       = DRIVER_NAME "/input0";
	input_dev->id.bustype = BUS_HOST;
	input_dev->id.vendor  = 0;
	input_dev->dev.parent = &lis3_dev.pdev->dev;
P
Pavel Machek 已提交
507

E
Eric Piel 已提交
508
	set_bit(EV_ABS, input_dev->evbit);
S
Samu Onkalo 已提交
509 510 511 512 513 514
	max_val = (lis3_dev.mdps_max_val * lis3_dev.scale) / LIS3_ACCURACY;
	fuzz = (LIS3_DEFAULT_FUZZ * lis3_dev.scale) / LIS3_ACCURACY;
	flat = (LIS3_DEFAULT_FLAT * lis3_dev.scale) / LIS3_ACCURACY;
	input_set_abs_params(input_dev, ABS_X, -max_val, max_val, fuzz, flat);
	input_set_abs_params(input_dev, ABS_Y, -max_val, max_val, fuzz, flat);
	input_set_abs_params(input_dev, ABS_Z, -max_val, max_val, fuzz, flat);
P
Pavel Machek 已提交
515

516 517 518 519
	lis3_dev.mapped_btns[0] = lis3lv02d_get_axis(abs(lis3_dev.ac.x), btns);
	lis3_dev.mapped_btns[1] = lis3lv02d_get_axis(abs(lis3_dev.ac.y), btns);
	lis3_dev.mapped_btns[2] = lis3lv02d_get_axis(abs(lis3_dev.ac.z), btns);

E
Eric Piel 已提交
520
	err = input_register_polled_device(lis3_dev.idev);
P
Pavel Machek 已提交
521
	if (err) {
E
Eric Piel 已提交
522
		input_free_polled_device(lis3_dev.idev);
523
		lis3_dev.idev = NULL;
P
Pavel Machek 已提交
524 525 526 527
	}

	return err;
}
528
EXPORT_SYMBOL_GPL(lis3lv02d_joystick_enable);
P
Pavel Machek 已提交
529

530
void lis3lv02d_joystick_disable(void)
P
Pavel Machek 已提交
531
{
532 533 534 535 536
	if (lis3_dev.irq)
		free_irq(lis3_dev.irq, &lis3_dev);
	if (lis3_dev.pdata && lis3_dev.pdata->irq2)
		free_irq(lis3_dev.pdata->irq2, &lis3_dev);

537
	if (!lis3_dev.idev)
P
Pavel Machek 已提交
538 539
		return;

540 541
	if (lis3_dev.irq)
		misc_deregister(&lis3lv02d_misc_device);
E
Eric Piel 已提交
542
	input_unregister_polled_device(lis3_dev.idev);
543
	input_free_polled_device(lis3_dev.idev);
544
	lis3_dev.idev = NULL;
P
Pavel Machek 已提交
545
}
546
EXPORT_SYMBOL_GPL(lis3lv02d_joystick_disable);
P
Pavel Machek 已提交
547 548

/* Sysfs stuff */
S
Samu Onkalo 已提交
549 550 551 552 553 554 555 556 557 558 559
static ssize_t lis3lv02d_selftest_show(struct device *dev,
				struct device_attribute *attr, char *buf)
{
	int result;
	s16 values[3];

	result = lis3lv02d_selftest(&lis3_dev, values);
	return sprintf(buf, "%s %d %d %d\n", result == 0 ? "OK" : "FAIL",
		values[0], values[1], values[2]);
}

P
Pavel Machek 已提交
560 561 562 563 564
static ssize_t lis3lv02d_position_show(struct device *dev,
				struct device_attribute *attr, char *buf)
{
	int x, y, z;

565
	mutex_lock(&lis3_dev.mutex);
566
	lis3lv02d_get_xyz(&lis3_dev, &x, &y, &z);
567
	mutex_unlock(&lis3_dev.mutex);
P
Pavel Machek 已提交
568 569 570 571 572 573
	return sprintf(buf, "(%d,%d,%d)\n", x, y, z);
}

static ssize_t lis3lv02d_rate_show(struct device *dev,
			struct device_attribute *attr, char *buf)
{
S
Samu Onkalo 已提交
574
	return sprintf(buf, "%d\n", lis3lv02d_get_odr());
P
Pavel Machek 已提交
575 576
}

577 578 579 580 581 582 583 584 585 586 587 588 589 590 591
static ssize_t lis3lv02d_rate_set(struct device *dev,
				struct device_attribute *attr, const char *buf,
				size_t count)
{
	unsigned long rate;

	if (strict_strtoul(buf, 0, &rate))
		return -EINVAL;

	if (lis3lv02d_set_odr(rate))
		return -EINVAL;

	return count;
}

S
Samu Onkalo 已提交
592
static DEVICE_ATTR(selftest, S_IRUSR, lis3lv02d_selftest_show, NULL);
P
Pavel Machek 已提交
593
static DEVICE_ATTR(position, S_IRUGO, lis3lv02d_position_show, NULL);
594 595
static DEVICE_ATTR(rate, S_IRUGO | S_IWUSR, lis3lv02d_rate_show,
					    lis3lv02d_rate_set);
P
Pavel Machek 已提交
596 597

static struct attribute *lis3lv02d_attributes[] = {
S
Samu Onkalo 已提交
598
	&dev_attr_selftest.attr,
P
Pavel Machek 已提交
599 600 601 602 603 604 605 606 607
	&dev_attr_position.attr,
	&dev_attr_rate.attr,
	NULL
};

static struct attribute_group lis3lv02d_attribute_group = {
	.attrs = lis3lv02d_attributes
};

608

609
static int lis3lv02d_add_fs(struct lis3lv02d *lis3)
P
Pavel Machek 已提交
610
{
611 612 613
	lis3->pdev = platform_device_register_simple(DRIVER_NAME, -1, NULL, 0);
	if (IS_ERR(lis3->pdev))
		return PTR_ERR(lis3->pdev);
P
Pavel Machek 已提交
614

615
	return sysfs_create_group(&lis3->pdev->dev.kobj, &lis3lv02d_attribute_group);
P
Pavel Machek 已提交
616 617
}

618
int lis3lv02d_remove_fs(struct lis3lv02d *lis3)
P
Pavel Machek 已提交
619
{
620 621
	sysfs_remove_group(&lis3->pdev->dev.kobj, &lis3lv02d_attribute_group);
	platform_device_unregister(lis3->pdev);
P
Pavel Machek 已提交
622 623
	return 0;
}
624
EXPORT_SYMBOL_GPL(lis3lv02d_remove_fs);
P
Pavel Machek 已提交
625

626 627 628
static void lis3lv02d_8b_configure(struct lis3lv02d *dev,
				struct lis3lv02d_platform_data *p)
{
629
	int err;
630 631
	int ctrl2 = p->hipass_ctrl;

632 633 634 635 636 637 638 639 640
	if (p->click_flags) {
		dev->write(dev, CLICK_CFG, p->click_flags);
		dev->write(dev, CLICK_TIMELIMIT, p->click_time_limit);
		dev->write(dev, CLICK_LATENCY, p->click_latency);
		dev->write(dev, CLICK_WINDOW, p->click_window);
		dev->write(dev, CLICK_THSZ, p->click_thresh_z & 0xf);
		dev->write(dev, CLICK_THSY_X,
			(p->click_thresh_x & 0xf) |
			(p->click_thresh_y << 4));
641 642 643 644 645 646 647

		if (dev->idev) {
			struct input_dev *input_dev = lis3_dev.idev->input;
			input_set_capability(input_dev, EV_KEY, BTN_X);
			input_set_capability(input_dev, EV_KEY, BTN_Y);
			input_set_capability(input_dev, EV_KEY, BTN_Z);
		}
648 649 650 651 652 653 654
	}

	if (p->wakeup_flags) {
		dev->write(dev, FF_WU_CFG_1, p->wakeup_flags);
		dev->write(dev, FF_WU_THS_1, p->wakeup_thresh & 0x7f);
		/* default to 2.5ms for now */
		dev->write(dev, FF_WU_DURATION_1, 1);
655 656 657 658 659 660 661 662 663
		ctrl2 ^= HP_FF_WU1; /* Xor to keep compatible with old pdata*/
	}

	if (p->wakeup_flags2) {
		dev->write(dev, FF_WU_CFG_2, p->wakeup_flags2);
		dev->write(dev, FF_WU_THS_2, p->wakeup_thresh2 & 0x7f);
		/* default to 2.5ms for now */
		dev->write(dev, FF_WU_DURATION_2, 1);
		ctrl2 ^= HP_FF_WU2; /* Xor to keep compatible with old pdata*/
664
	}
665 666
	/* Configure hipass filters */
	dev->write(dev, CTRL_REG2, ctrl2);
667 668 669 670 671 672 673 674 675 676 677 678

	if (p->irq2) {
		err = request_threaded_irq(p->irq2,
					NULL,
					lis302dl_interrupt_thread2_8b,
					IRQF_TRIGGER_RISING |
					IRQF_ONESHOT,
					DRIVER_NAME, &lis3_dev);
		if (err < 0)
			printk(KERN_ERR DRIVER_NAME
				"No second IRQ. Limited functionality\n");
	}
679 680
}

681 682
/*
 * Initialise the accelerometer and the various subsystems.
683
 * Should be rather independent of the bus system.
684
 */
685
int lis3lv02d_init_device(struct lis3lv02d *dev)
686
{
687 688 689
	int err;
	irq_handler_t thread_fn;

690 691 692
	dev->whoami = lis3lv02d_read_8(dev, WHO_AM_I);

	switch (dev->whoami) {
693 694 695
	case WAI_12B:
		printk(KERN_INFO DRIVER_NAME ": 12 bits sensor found\n");
		dev->read_data = lis3lv02d_read_12;
696
		dev->mdps_max_val = 2048;
S
Samu Onkalo 已提交
697
		dev->pwron_delay = LIS3_PWRON_DELAY_WAI_12B;
698 699
		dev->odrs = lis3_12_rates;
		dev->odr_mask = CTRL1_DF0 | CTRL1_DF1;
S
Samu Onkalo 已提交
700
		dev->scale = LIS3_SENSITIVITY_12B;
701
		break;
702 703
	case WAI_8B:
		printk(KERN_INFO DRIVER_NAME ": 8 bits sensor found\n");
704 705
		dev->read_data = lis3lv02d_read_8;
		dev->mdps_max_val = 128;
S
Samu Onkalo 已提交
706
		dev->pwron_delay = LIS3_PWRON_DELAY_WAI_8B;
707 708
		dev->odrs = lis3_8_rates;
		dev->odr_mask = CTRL1_DR;
S
Samu Onkalo 已提交
709
		dev->scale = LIS3_SENSITIVITY_8B;
710
		break;
711 712 713 714 715 716 717 718 719
	case WAI_3DC:
		printk(KERN_INFO DRIVER_NAME ": 8 bits 3DC sensor found\n");
		dev->read_data = lis3lv02d_read_8;
		dev->mdps_max_val = 128;
		dev->pwron_delay = LIS3_PWRON_DELAY_WAI_8B;
		dev->odrs = lis3_3dc_rates;
		dev->odr_mask = CTRL1_ODR0|CTRL1_ODR1|CTRL1_ODR2|CTRL1_ODR3;
		dev->scale = LIS3_SENSITIVITY_8B;
		break;
720 721
	default:
		printk(KERN_ERR DRIVER_NAME
722
			": unknown sensor type 0x%X\n", dev->whoami);
723 724 725
		return -EINVAL;
	}

S
Samu Onkalo 已提交
726 727
	mutex_init(&dev->mutex);

728
	lis3lv02d_add_fs(dev);
729
	lis3lv02d_poweron(dev);
730 731 732 733

	if (lis3lv02d_joystick_enable())
		printk(KERN_ERR DRIVER_NAME ": joystick initialization failed\n");

D
Daniel Mack 已提交
734 735 736 737 738
	/* passing in platform specific data is purely optional and only
	 * used by the SPI transport layer at the moment */
	if (dev->pdata) {
		struct lis3lv02d_platform_data *p = dev->pdata;

739 740
		if (dev->whoami == WAI_8B)
			lis3lv02d_8b_configure(dev, p);
741

D
Daniel Mack 已提交
742 743 744 745
		if (p->irq_cfg)
			dev->write(dev, CTRL_REG3, p->irq_cfg);
	}

746
	/* bail if we did not get an IRQ from the bus layer */
747 748
	if (!dev->irq) {
		printk(KERN_ERR DRIVER_NAME
749
			": No IRQ. Disabling /dev/freefall\n");
750 751 752
		goto out;
	}

753 754 755 756 757 758 759 760 761 762 763
	/*
	 * The sensor can generate interrupts for free-fall and direction
	 * detection (distinguishable with FF_WU_SRC and DD_SRC) but to keep
	 * the things simple and _fast_ we activate it only for free-fall, so
	 * no need to read register (very slow with ACPI). For the same reason,
	 * we forbid shared interrupts.
	 *
	 * IRQF_TRIGGER_RISING seems pointless on HP laptops because the
	 * io-apic is not configurable (and generates a warning) but I keep it
	 * in case of support for other hardware.
	 */
764
	if (dev->pdata && dev->whoami == WAI_8B)
765 766 767 768 769 770 771 772 773 774 775 776 777 778
		thread_fn = lis302dl_interrupt_thread1_8b;
	else
		thread_fn = NULL;

	err = request_threaded_irq(dev->irq, lis302dl_interrupt,
				thread_fn,
				IRQF_TRIGGER_RISING | IRQF_ONESHOT,
				DRIVER_NAME, &lis3_dev);

	if (err < 0) {
		printk(KERN_ERR DRIVER_NAME "Cannot get IRQ\n");
		goto out;
	}

779 780 781 782 783 784 785
	if (misc_register(&lis3lv02d_misc_device))
		printk(KERN_ERR DRIVER_NAME ": misc_register failed\n");
out:
	return 0;
}
EXPORT_SYMBOL_GPL(lis3lv02d_init_device);

P
Pavel Machek 已提交
786
MODULE_DESCRIPTION("ST LIS3LV02Dx three-axis digital accelerometer driver");
787
MODULE_AUTHOR("Yan Burman, Eric Piel, Pavel Machek");
P
Pavel Machek 已提交
788
MODULE_LICENSE("GPL");