Path: blob/master/drivers/input/keyboard/mpr121_touchkey.c
15111 views
/*1* Touchkey driver for Freescale MPR121 Controllor2*3* Copyright (C) 2011 Freescale Semiconductor, Inc.4* Author: Zhang Jiejing <[email protected]>5*6* Based on mcs_touchkey.c7*8* This program is free software; you can redistribute it and/or modify9* it under the terms of the GNU General Public License version 2 as10* published by the Free Software Foundation.11*12*/1314#include <linux/module.h>15#include <linux/init.h>16#include <linux/input.h>17#include <linux/i2c.h>18#include <linux/slab.h>19#include <linux/delay.h>20#include <linux/bitops.h>21#include <linux/interrupt.h>22#include <linux/i2c/mpr121_touchkey.h>2324/* Register definitions */25#define ELE_TOUCH_STATUS_0_ADDR 0x026#define ELE_TOUCH_STATUS_1_ADDR 0X127#define MHD_RISING_ADDR 0x2b28#define NHD_RISING_ADDR 0x2c29#define NCL_RISING_ADDR 0x2d30#define FDL_RISING_ADDR 0x2e31#define MHD_FALLING_ADDR 0x2f32#define NHD_FALLING_ADDR 0x3033#define NCL_FALLING_ADDR 0x3134#define FDL_FALLING_ADDR 0x3235#define ELE0_TOUCH_THRESHOLD_ADDR 0x4136#define ELE0_RELEASE_THRESHOLD_ADDR 0x4237#define AFE_CONF_ADDR 0x5c38#define FILTER_CONF_ADDR 0x5d3940/*41* ELECTRODE_CONF_ADDR: This register configures the number of42* enabled capacitance sensing inputs and its run/suspend mode.43*/44#define ELECTRODE_CONF_ADDR 0x5e45#define AUTO_CONFIG_CTRL_ADDR 0x7b46#define AUTO_CONFIG_USL_ADDR 0x7d47#define AUTO_CONFIG_LSL_ADDR 0x7e48#define AUTO_CONFIG_TL_ADDR 0x7f4950/* Threshold of touch/release trigger */51#define TOUCH_THRESHOLD 0x0f52#define RELEASE_THRESHOLD 0x0a53/* Masks for touch and release triggers */54#define TOUCH_STATUS_MASK 0xfff55/* MPR121 has 12 keys */56#define MPR121_MAX_KEY_COUNT 125758struct mpr121_touchkey {59struct i2c_client *client;60struct input_dev *input_dev;61unsigned int key_val;62unsigned int statusbits;63unsigned int keycount;64u16 keycodes[MPR121_MAX_KEY_COUNT];65};6667struct mpr121_init_register {68int addr;69u8 val;70};7172static const struct mpr121_init_register init_reg_table[] __devinitconst = {73{ MHD_RISING_ADDR, 0x1 },74{ NHD_RISING_ADDR, 0x1 },75{ MHD_FALLING_ADDR, 0x1 },76{ NHD_FALLING_ADDR, 0x1 },77{ NCL_FALLING_ADDR, 0xff },78{ FDL_FALLING_ADDR, 0x02 },79{ FILTER_CONF_ADDR, 0x04 },80{ AFE_CONF_ADDR, 0x0b },81{ AUTO_CONFIG_CTRL_ADDR, 0x0b },82};8384static irqreturn_t mpr_touchkey_interrupt(int irq, void *dev_id)85{86struct mpr121_touchkey *mpr121 = dev_id;87struct i2c_client *client = mpr121->client;88struct input_dev *input = mpr121->input_dev;89unsigned int key_num, key_val, pressed;90int reg;9192reg = i2c_smbus_read_byte_data(client, ELE_TOUCH_STATUS_1_ADDR);93if (reg < 0) {94dev_err(&client->dev, "i2c read error [%d]\n", reg);95goto out;96}9798reg <<= 8;99reg |= i2c_smbus_read_byte_data(client, ELE_TOUCH_STATUS_0_ADDR);100if (reg < 0) {101dev_err(&client->dev, "i2c read error [%d]\n", reg);102goto out;103}104105reg &= TOUCH_STATUS_MASK;106/* use old press bit to figure out which bit changed */107key_num = ffs(reg ^ mpr121->statusbits) - 1;108pressed = reg & (1 << key_num);109mpr121->statusbits = reg;110111key_val = mpr121->keycodes[key_num];112113input_event(input, EV_MSC, MSC_SCAN, key_num);114input_report_key(input, key_val, pressed);115input_sync(input);116117dev_dbg(&client->dev, "key %d %d %s\n", key_num, key_val,118pressed ? "pressed" : "released");119120out:121return IRQ_HANDLED;122}123124static int __devinit mpr121_phys_init(const struct mpr121_platform_data *pdata,125struct mpr121_touchkey *mpr121,126struct i2c_client *client)127{128const struct mpr121_init_register *reg;129unsigned char usl, lsl, tl;130int i, t, vdd, ret;131132/* Set up touch/release threshold for ele0-ele11 */133for (i = 0; i <= MPR121_MAX_KEY_COUNT; i++) {134t = ELE0_TOUCH_THRESHOLD_ADDR + (i * 2);135ret = i2c_smbus_write_byte_data(client, t, TOUCH_THRESHOLD);136if (ret < 0)137goto err_i2c_write;138ret = i2c_smbus_write_byte_data(client, t + 1,139RELEASE_THRESHOLD);140if (ret < 0)141goto err_i2c_write;142}143144/* Set up init register */145for (i = 0; i < ARRAY_SIZE(init_reg_table); i++) {146reg = &init_reg_table[i];147ret = i2c_smbus_write_byte_data(client, reg->addr, reg->val);148if (ret < 0)149goto err_i2c_write;150}151152153/*154* Capacitance on sensing input varies and needs to be compensated.155* The internal MPR121-auto-configuration can do this if it's156* registers are set properly (based on pdata->vdd_uv).157*/158vdd = pdata->vdd_uv / 1000;159usl = ((vdd - 700) * 256) / vdd;160lsl = (usl * 65) / 100;161tl = (usl * 90) / 100;162ret = i2c_smbus_write_byte_data(client, AUTO_CONFIG_USL_ADDR, usl);163ret |= i2c_smbus_write_byte_data(client, AUTO_CONFIG_LSL_ADDR, lsl);164ret |= i2c_smbus_write_byte_data(client, AUTO_CONFIG_TL_ADDR, tl);165ret |= i2c_smbus_write_byte_data(client, ELECTRODE_CONF_ADDR,166mpr121->keycount);167if (ret != 0)168goto err_i2c_write;169170dev_dbg(&client->dev, "set up with %x keys.\n", mpr121->keycount);171172return 0;173174err_i2c_write:175dev_err(&client->dev, "i2c write error: %d\n", ret);176return ret;177}178179static int __devinit mpr_touchkey_probe(struct i2c_client *client,180const struct i2c_device_id *id)181{182const struct mpr121_platform_data *pdata = client->dev.platform_data;183struct mpr121_touchkey *mpr121;184struct input_dev *input_dev;185int error;186int i;187188if (!pdata) {189dev_err(&client->dev, "no platform data defined\n");190return -EINVAL;191}192193if (!pdata->keymap || !pdata->keymap_size) {194dev_err(&client->dev, "missing keymap data\n");195return -EINVAL;196}197198if (pdata->keymap_size > MPR121_MAX_KEY_COUNT) {199dev_err(&client->dev, "too many keys defined\n");200return -EINVAL;201}202203if (!client->irq) {204dev_err(&client->dev, "irq number should not be zero\n");205return -EINVAL;206}207208mpr121 = kzalloc(sizeof(struct mpr121_touchkey), GFP_KERNEL);209input_dev = input_allocate_device();210if (!mpr121 || !input_dev) {211dev_err(&client->dev, "Failed to allocate memory\n");212error = -ENOMEM;213goto err_free_mem;214}215216mpr121->client = client;217mpr121->input_dev = input_dev;218mpr121->keycount = pdata->keymap_size;219220input_dev->name = "Freescale MPR121 Touchkey";221input_dev->id.bustype = BUS_I2C;222input_dev->dev.parent = &client->dev;223input_dev->evbit[0] = BIT_MASK(EV_KEY) | BIT_MASK(EV_REP);224225input_dev->keycode = mpr121->keycodes;226input_dev->keycodesize = sizeof(mpr121->keycodes[0]);227input_dev->keycodemax = mpr121->keycount;228229for (i = 0; i < pdata->keymap_size; i++) {230input_set_capability(input_dev, EV_KEY, pdata->keymap[i]);231mpr121->keycodes[i] = pdata->keymap[i];232}233234error = mpr121_phys_init(pdata, mpr121, client);235if (error) {236dev_err(&client->dev, "Failed to init register\n");237goto err_free_mem;238}239240error = request_threaded_irq(client->irq, NULL,241mpr_touchkey_interrupt,242IRQF_TRIGGER_FALLING,243client->dev.driver->name, mpr121);244if (error) {245dev_err(&client->dev, "Failed to register interrupt\n");246goto err_free_mem;247}248249error = input_register_device(input_dev);250if (error)251goto err_free_irq;252253i2c_set_clientdata(client, mpr121);254device_init_wakeup(&client->dev, pdata->wakeup);255256return 0;257258err_free_irq:259free_irq(client->irq, mpr121);260err_free_mem:261input_free_device(input_dev);262kfree(mpr121);263return error;264}265266static int __devexit mpr_touchkey_remove(struct i2c_client *client)267{268struct mpr121_touchkey *mpr121 = i2c_get_clientdata(client);269270free_irq(client->irq, mpr121);271input_unregister_device(mpr121->input_dev);272kfree(mpr121);273274return 0;275}276277#ifdef CONFIG_PM_SLEEP278static int mpr_suspend(struct device *dev)279{280struct i2c_client *client = to_i2c_client(dev);281282if (device_may_wakeup(&client->dev))283enable_irq_wake(client->irq);284285i2c_smbus_write_byte_data(client, ELECTRODE_CONF_ADDR, 0x00);286287return 0;288}289290static int mpr_resume(struct device *dev)291{292struct i2c_client *client = to_i2c_client(dev);293struct mpr121_touchkey *mpr121 = i2c_get_clientdata(client);294295if (device_may_wakeup(&client->dev))296disable_irq_wake(client->irq);297298i2c_smbus_write_byte_data(client, ELECTRODE_CONF_ADDR,299mpr121->keycount);300301return 0;302}303#endif304305static SIMPLE_DEV_PM_OPS(mpr121_touchkey_pm_ops, mpr_suspend, mpr_resume);306307static const struct i2c_device_id mpr121_id[] = {308{ "mpr121_touchkey", 0 },309{ }310};311MODULE_DEVICE_TABLE(i2c, mpr121_id);312313static struct i2c_driver mpr_touchkey_driver = {314.driver = {315.name = "mpr121",316.owner = THIS_MODULE,317.pm = &mpr121_touchkey_pm_ops,318},319.id_table = mpr121_id,320.probe = mpr_touchkey_probe,321.remove = __devexit_p(mpr_touchkey_remove),322};323324static int __init mpr_touchkey_init(void)325{326return i2c_add_driver(&mpr_touchkey_driver);327}328module_init(mpr_touchkey_init);329330static void __exit mpr_touchkey_exit(void)331{332i2c_del_driver(&mpr_touchkey_driver);333}334module_exit(mpr_touchkey_exit);335336MODULE_LICENSE("GPL");337MODULE_AUTHOR("Zhang Jiejing <[email protected]>");338MODULE_DESCRIPTION("Touch Key driver for Freescale MPR121 Chip");339340341