162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Driver for Freescale's 3-Axis Accelerometer MMA8450 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (C) 2011 Freescale Semiconductor, Inc. All Rights Reserved. 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include <linux/kernel.h> 962306a36Sopenharmony_ci#include <linux/module.h> 1062306a36Sopenharmony_ci#include <linux/slab.h> 1162306a36Sopenharmony_ci#include <linux/delay.h> 1262306a36Sopenharmony_ci#include <linux/i2c.h> 1362306a36Sopenharmony_ci#include <linux/input.h> 1462306a36Sopenharmony_ci#include <linux/mod_devicetable.h> 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ci#define MMA8450_DRV_NAME "mma8450" 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#define MODE_CHANGE_DELAY_MS 100 1962306a36Sopenharmony_ci#define POLL_INTERVAL 100 2062306a36Sopenharmony_ci#define POLL_INTERVAL_MAX 500 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci/* register definitions */ 2362306a36Sopenharmony_ci#define MMA8450_STATUS 0x00 2462306a36Sopenharmony_ci#define MMA8450_STATUS_ZXYDR 0x08 2562306a36Sopenharmony_ci 2662306a36Sopenharmony_ci#define MMA8450_OUT_X8 0x01 2762306a36Sopenharmony_ci#define MMA8450_OUT_Y8 0x02 2862306a36Sopenharmony_ci#define MMA8450_OUT_Z8 0x03 2962306a36Sopenharmony_ci 3062306a36Sopenharmony_ci#define MMA8450_OUT_X_LSB 0x05 3162306a36Sopenharmony_ci#define MMA8450_OUT_X_MSB 0x06 3262306a36Sopenharmony_ci#define MMA8450_OUT_Y_LSB 0x07 3362306a36Sopenharmony_ci#define MMA8450_OUT_Y_MSB 0x08 3462306a36Sopenharmony_ci#define MMA8450_OUT_Z_LSB 0x09 3562306a36Sopenharmony_ci#define MMA8450_OUT_Z_MSB 0x0a 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_ci#define MMA8450_XYZ_DATA_CFG 0x16 3862306a36Sopenharmony_ci 3962306a36Sopenharmony_ci#define MMA8450_CTRL_REG1 0x38 4062306a36Sopenharmony_ci#define MMA8450_CTRL_REG2 0x39 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_cistatic int mma8450_read(struct i2c_client *c, unsigned int off) 4362306a36Sopenharmony_ci{ 4462306a36Sopenharmony_ci int ret; 4562306a36Sopenharmony_ci 4662306a36Sopenharmony_ci ret = i2c_smbus_read_byte_data(c, off); 4762306a36Sopenharmony_ci if (ret < 0) 4862306a36Sopenharmony_ci dev_err(&c->dev, 4962306a36Sopenharmony_ci "failed to read register 0x%02x, error %d\n", 5062306a36Sopenharmony_ci off, ret); 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci return ret; 5362306a36Sopenharmony_ci} 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_cistatic int mma8450_write(struct i2c_client *c, unsigned int off, u8 v) 5662306a36Sopenharmony_ci{ 5762306a36Sopenharmony_ci int error; 5862306a36Sopenharmony_ci 5962306a36Sopenharmony_ci error = i2c_smbus_write_byte_data(c, off, v); 6062306a36Sopenharmony_ci if (error < 0) { 6162306a36Sopenharmony_ci dev_err(&c->dev, 6262306a36Sopenharmony_ci "failed to write to register 0x%02x, error %d\n", 6362306a36Sopenharmony_ci off, error); 6462306a36Sopenharmony_ci return error; 6562306a36Sopenharmony_ci } 6662306a36Sopenharmony_ci 6762306a36Sopenharmony_ci return 0; 6862306a36Sopenharmony_ci} 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_cistatic int mma8450_read_block(struct i2c_client *c, unsigned int off, 7162306a36Sopenharmony_ci u8 *buf, size_t size) 7262306a36Sopenharmony_ci{ 7362306a36Sopenharmony_ci int err; 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci err = i2c_smbus_read_i2c_block_data(c, off, size, buf); 7662306a36Sopenharmony_ci if (err < 0) { 7762306a36Sopenharmony_ci dev_err(&c->dev, 7862306a36Sopenharmony_ci "failed to read block data at 0x%02x, error %d\n", 7962306a36Sopenharmony_ci MMA8450_OUT_X_LSB, err); 8062306a36Sopenharmony_ci return err; 8162306a36Sopenharmony_ci } 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci return 0; 8462306a36Sopenharmony_ci} 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_cistatic void mma8450_poll(struct input_dev *input) 8762306a36Sopenharmony_ci{ 8862306a36Sopenharmony_ci struct i2c_client *c = input_get_drvdata(input); 8962306a36Sopenharmony_ci int x, y, z; 9062306a36Sopenharmony_ci int ret; 9162306a36Sopenharmony_ci u8 buf[6]; 9262306a36Sopenharmony_ci 9362306a36Sopenharmony_ci ret = mma8450_read(c, MMA8450_STATUS); 9462306a36Sopenharmony_ci if (ret < 0) 9562306a36Sopenharmony_ci return; 9662306a36Sopenharmony_ci 9762306a36Sopenharmony_ci if (!(ret & MMA8450_STATUS_ZXYDR)) 9862306a36Sopenharmony_ci return; 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci ret = mma8450_read_block(c, MMA8450_OUT_X_LSB, buf, sizeof(buf)); 10162306a36Sopenharmony_ci if (ret < 0) 10262306a36Sopenharmony_ci return; 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_ci x = ((int)(s8)buf[1] << 4) | (buf[0] & 0xf); 10562306a36Sopenharmony_ci y = ((int)(s8)buf[3] << 4) | (buf[2] & 0xf); 10662306a36Sopenharmony_ci z = ((int)(s8)buf[5] << 4) | (buf[4] & 0xf); 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_ci input_report_abs(input, ABS_X, x); 10962306a36Sopenharmony_ci input_report_abs(input, ABS_Y, y); 11062306a36Sopenharmony_ci input_report_abs(input, ABS_Z, z); 11162306a36Sopenharmony_ci input_sync(input); 11262306a36Sopenharmony_ci} 11362306a36Sopenharmony_ci 11462306a36Sopenharmony_ci/* Initialize the MMA8450 chip */ 11562306a36Sopenharmony_cistatic int mma8450_open(struct input_dev *input) 11662306a36Sopenharmony_ci{ 11762306a36Sopenharmony_ci struct i2c_client *c = input_get_drvdata(input); 11862306a36Sopenharmony_ci int err; 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_ci /* enable all events from X/Y/Z, no FIFO */ 12162306a36Sopenharmony_ci err = mma8450_write(c, MMA8450_XYZ_DATA_CFG, 0x07); 12262306a36Sopenharmony_ci if (err) 12362306a36Sopenharmony_ci return err; 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci /* 12662306a36Sopenharmony_ci * Sleep mode poll rate - 50Hz 12762306a36Sopenharmony_ci * System output data rate - 400Hz 12862306a36Sopenharmony_ci * Full scale selection - Active, +/- 2G 12962306a36Sopenharmony_ci */ 13062306a36Sopenharmony_ci err = mma8450_write(c, MMA8450_CTRL_REG1, 0x01); 13162306a36Sopenharmony_ci if (err) 13262306a36Sopenharmony_ci return err; 13362306a36Sopenharmony_ci 13462306a36Sopenharmony_ci msleep(MODE_CHANGE_DELAY_MS); 13562306a36Sopenharmony_ci return 0; 13662306a36Sopenharmony_ci} 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_cistatic void mma8450_close(struct input_dev *input) 13962306a36Sopenharmony_ci{ 14062306a36Sopenharmony_ci struct i2c_client *c = input_get_drvdata(input); 14162306a36Sopenharmony_ci 14262306a36Sopenharmony_ci mma8450_write(c, MMA8450_CTRL_REG1, 0x00); 14362306a36Sopenharmony_ci mma8450_write(c, MMA8450_CTRL_REG2, 0x01); 14462306a36Sopenharmony_ci} 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci/* 14762306a36Sopenharmony_ci * I2C init/probing/exit functions 14862306a36Sopenharmony_ci */ 14962306a36Sopenharmony_cistatic int mma8450_probe(struct i2c_client *c) 15062306a36Sopenharmony_ci{ 15162306a36Sopenharmony_ci struct input_dev *input; 15262306a36Sopenharmony_ci int err; 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci input = devm_input_allocate_device(&c->dev); 15562306a36Sopenharmony_ci if (!input) 15662306a36Sopenharmony_ci return -ENOMEM; 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_ci input_set_drvdata(input, c); 15962306a36Sopenharmony_ci 16062306a36Sopenharmony_ci input->name = MMA8450_DRV_NAME; 16162306a36Sopenharmony_ci input->id.bustype = BUS_I2C; 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci input->open = mma8450_open; 16462306a36Sopenharmony_ci input->close = mma8450_close; 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci input_set_abs_params(input, ABS_X, -2048, 2047, 32, 32); 16762306a36Sopenharmony_ci input_set_abs_params(input, ABS_Y, -2048, 2047, 32, 32); 16862306a36Sopenharmony_ci input_set_abs_params(input, ABS_Z, -2048, 2047, 32, 32); 16962306a36Sopenharmony_ci 17062306a36Sopenharmony_ci err = input_setup_polling(input, mma8450_poll); 17162306a36Sopenharmony_ci if (err) { 17262306a36Sopenharmony_ci dev_err(&c->dev, "failed to set up polling\n"); 17362306a36Sopenharmony_ci return err; 17462306a36Sopenharmony_ci } 17562306a36Sopenharmony_ci 17662306a36Sopenharmony_ci input_set_poll_interval(input, POLL_INTERVAL); 17762306a36Sopenharmony_ci input_set_max_poll_interval(input, POLL_INTERVAL_MAX); 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci err = input_register_device(input); 18062306a36Sopenharmony_ci if (err) { 18162306a36Sopenharmony_ci dev_err(&c->dev, "failed to register input device\n"); 18262306a36Sopenharmony_ci return err; 18362306a36Sopenharmony_ci } 18462306a36Sopenharmony_ci 18562306a36Sopenharmony_ci return 0; 18662306a36Sopenharmony_ci} 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_cistatic const struct i2c_device_id mma8450_id[] = { 18962306a36Sopenharmony_ci { MMA8450_DRV_NAME, 0 }, 19062306a36Sopenharmony_ci { }, 19162306a36Sopenharmony_ci}; 19262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(i2c, mma8450_id); 19362306a36Sopenharmony_ci 19462306a36Sopenharmony_cistatic const struct of_device_id mma8450_dt_ids[] = { 19562306a36Sopenharmony_ci { .compatible = "fsl,mma8450", }, 19662306a36Sopenharmony_ci { /* sentinel */ } 19762306a36Sopenharmony_ci}; 19862306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, mma8450_dt_ids); 19962306a36Sopenharmony_ci 20062306a36Sopenharmony_cistatic struct i2c_driver mma8450_driver = { 20162306a36Sopenharmony_ci .driver = { 20262306a36Sopenharmony_ci .name = MMA8450_DRV_NAME, 20362306a36Sopenharmony_ci .of_match_table = mma8450_dt_ids, 20462306a36Sopenharmony_ci }, 20562306a36Sopenharmony_ci .probe = mma8450_probe, 20662306a36Sopenharmony_ci .id_table = mma8450_id, 20762306a36Sopenharmony_ci}; 20862306a36Sopenharmony_ci 20962306a36Sopenharmony_cimodule_i2c_driver(mma8450_driver); 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ciMODULE_AUTHOR("Freescale Semiconductor, Inc."); 21262306a36Sopenharmony_ciMODULE_DESCRIPTION("MMA8450 3-Axis Accelerometer Driver"); 21362306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 214