Hi Tobita, On Mon, Nov 14, 2011 at 05:51:18PM +0900, 飛田龍之介 wrote: > Hi all > > I hope this mail will work. > Please help to review wacom_i2c driver for wacom stylus with I2C interface. It is in my queue to review, sorry for the delay... > Thanks! > Tats > > 2011/11/14 Tobita Tatsunosuke <tobita.tatsunosuke@xxxxxxxxxxx> > > > > > > -----Original Message----- > > From: Tobita Tatsunosuke [mailto:tobita.tatsunosuke@xxxxxxxxxxx] > > Sent: Friday, November 04, 2011 9:44 PM > > To: 'linux-input@xxxxxxxxxxxxxxx' > > Cc: 'Dmitry Torokhov'; 'Ping Cheng'; 'Jason Gerecke' > > Subject: RE: [PATCH] Input:Add Wacom Stylus support for I2C > > > > Hello all > > > > It seems like my E-mail server didn't work well. > > I sent this again in case it doesn't really send the message. > > > > I posted two patches a week ago; one is for modification Kconfig and > > Makefile(attached txt) and the other is below the driver and the header > > files. > > However, I haven't gotten any comment and am even not sure if those files > > are going to be included in next release. > > Would you please help to review them? > > Thanks! > > > > Regards, > > > > Tats > > > > -----Original Message----- > > From: linux-input-owner@xxxxxxxxxxxxxxx > > [mailto:linux-input-owner@xxxxxxxxxxxxxxx] On Behalf Of Tatsunosuke Tobita > > Sent: Wednesday, October 19, 2011 3:36 PM > > To: linux-input@xxxxxxxxxxxxxxx > > Cc: Tatsunosuke Tobita > > Subject: [PATCH] Input:Add Wacom Stylus support for I2C > > > > From: Tatsunosuke Tobita <tobita.tatsunosuke@xxxxxxxxxxx> > > > > Adding support for Wacom Stylus with I2C interface. > > > > Signed-off-by: Tatsunosuke Tobita <tobita.tatsunosuke@xxxxxxxxxxx> > > --- > > drivers/input/touchscreen/wacom_i2c.c | 261 > > +++++++++++++++++++++++++++++++++ > > drivers/input/touchscreen/wacom_i2c.h | 72 +++++++++ > > 2 files changed, 333 insertions(+), 0 deletions(-) create mode 100644 > > drivers/input/touchscreen/wacom_i2c.c > > create mode 100644 drivers/input/touchscreen/wacom_i2c.h > > > > diff --git a/drivers/input/touchscreen/wacom_i2c.c > > b/drivers/input/touchscreen/wacom_i2c.c > > new file mode 100644 > > index 0000000..3f16f93 > > --- /dev/null > > +++ b/drivers/input/touchscreen/wacom_i2c.c > > @@ -0,0 +1,260 @@ > > +/* > > + * Wacom Penabled Driver for I2C > > + * > > + * Copyright (c) 2011 Tatsunosuke Tobita, Wacom. > > + * <tobita.tatsunosuke@xxxxxxxxxxx> > > + * > > + * 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 of 2 of the License, > > + * or (at your option) any later version. > > + */ > > + > > +#include "wacom_i2c.h" > > + > > +static int wacom_send_query(struct wacom_i2c *wac_i2c) { > > + int ret; > > + u8 cmd[4] = {CMD_QUERY0, CMD_QUERY1, CMD_QUERY2, CMD_QUERY3}; > > + u8 data[COM_COORD_NUM]; > > + > > + ret = i2c_master_send(wac_i2c->client, cmd, sizeof(cmd)); > > + if (ret < 0) { > > + dev_printk(KERN_ERR, &wac_i2c->client->dev, > > + "Sending 1st Query failed \n"); > > + return -1; > > + } > > + > > + cmd[0] = CMD_THROW0; cmd[1] = CMD_THROW1; > > + ret = i2c_master_send(wac_i2c->client, cmd, 2); > > + if (ret < 0) { > > + dev_printk(KERN_ERR, &wac_i2c->client->dev, > > + "Sending 2nd Query failed \n"); > > + return -1; > > + } > > + > > + ret = i2c_master_recv(wac_i2c->client, data, QUERY_SIZE); > > + if (ret < 0) { > > + dev_printk(KERN_ERR, &wac_i2c->client->dev, > > + "Receiving data failed 1\n"); > > + return -1; > > + } > > + > > + wac_i2c->wac_feature.x_max = get_unaligned_le16(&data[3]); > > + wac_i2c->wac_feature.y_max = get_unaligned_le16(&data[5]); > > + wac_i2c->wac_feature.pressure_max = get_unaligned_le16(&data[11]); > > + wac_i2c->wac_feature.fw_version = get_unaligned_le16(&data[13]); > > + > > + dev_printk(KERN_INFO, &wac_i2c->client->dev, > > + "x_max:%d, y_max:%d, pressure:%d, fw:%d\n", > > + wac_i2c->wac_feature.x_max, wac_i2c->wac_feature.y_max, > > + wac_i2c->wac_feature.pressure_max, > > +wac_i2c->wac_feature.fw_version); > > + > > + return 0; > > +} > > + > > +static void wacom_set_coordination(struct wacom_i2c *wac_i2c) { > > + int ret; > > + unsigned int x, y, pressure; > > + unsigned char tsw, f1, f2, ers; > > + u8 data[COM_COORD_NUM]; > > + > > + do { > > + ret = i2c_master_recv(wac_i2c->client, data, QUERY_SIZE); > > + } while (gpio_get_value(GPIO) <= 0); > > + > > + if (ret >= 0) { > > + tsw = data[3]&0x01; > > + ers = data[3]&0x04; > > + f1 = data[3]&0x02; > > + f2 = data[3]&0x10; > > + x = le16_to_cpup((__le16 *)&data[4]); > > + y = le16_to_cpup((__le16 *)&data[6]); > > + pressure = le16_to_cpup((__le16 *)&data[8]); > > + > > + input_report_abs(wac_i2c->input, ABS_X, x); > > + input_report_abs(wac_i2c->input, ABS_Y, y); > > + input_report_key(wac_i2c->input, BTN_TOOL_PEN, tsw); > > + input_report_key(wac_i2c->input, BTN_TOOL_RUBBER, ers); > > + input_report_abs(wac_i2c->input, ABS_PRESSURE, pressure); > > + input_report_key(wac_i2c->input, BTN_TOUCH, (tsw || ers)); > > + input_report_key(wac_i2c->input, BTN_STYLUS, f1); > > + input_report_key(wac_i2c->input, BTN_STYLUS2, f2); > > + > > + input_sync(wac_i2c->input); > > + } > > +} > > + > > +static irqreturn_t wacom_i2c_irq(int irqno, void *param) { > > + struct wacom_i2c *wac_i2c = param; > > + > > + wacom_set_coordination(wac_i2c); > > + > > + return IRQ_HANDLED; > > +} > > + > > + > > +static int wacom_i2c_input_open(struct input_dev *dev) { > > + struct wacom_i2c *wac_i2c = input_get_drvdata(dev); > > + int ret; > > + > > + __set_bit(ABS_X, wac_i2c->input->absbit); > > + __set_bit(ABS_Y, wac_i2c->input->absbit); > > + __set_bit(ABS_PRESSURE, wac_i2c->input->absbit); > > + __set_bit(BTN_TOOL_PEN, wac_i2c->input->keybit); > > + __set_bit(BTN_TOOL_RUBBER, wac_i2c->input->keybit); > > + __set_bit(BTN_STYLUS, wac_i2c->input->keybit); > > + __set_bit(BTN_STYLUS2, wac_i2c->input->keybit); > > + __set_bit(BTN_TOUCH, wac_i2c->input->keybit); > > + > > + ret = wacom_send_query(wac_i2c); > > + if (ret < 0) > > + return -EIO; > > + > > + wac_i2c->input->id.version = wac_i2c->wac_feature.fw_version; > > + input_set_abs_params(wac_i2c->input, ABS_X, 0, > > + wac_i2c->wac_feature.x_max, 0, 0); > > + input_set_abs_params(wac_i2c->input, ABS_Y, 0, > > + wac_i2c->wac_feature.y_max, 0, 0); > > + input_set_abs_params(wac_i2c->input, ABS_PRESSURE, 0, > > + wac_i2c->wac_feature.pressure_max, 0, 0); > > + > > + if (request_threaded_irq(EMR_IRQ, NULL, > > + wacom_i2c_irq, IRQF_TRIGGER_FALLING, > > "WACOM_I2C_IRQ", wac_i2c)) { > > + dev_printk(KERN_ERR, &wac_i2c->client->dev, "Failed to > > enable IRQ for EMR \n"); > > + return -EIO; > > + } > > + > > + /*Clear the buffer in the firmware*/ > > + wacom_set_coordination(wac_i2c); > > + > > + return 0; > > +} > > + > > +static void wacom_i2c_input_close(struct input_dev *dev) { > > + disable_irq(EMR_IRQ); > > +} > > + > > +static int wacom_i2c_probe(struct i2c_client *client, > > + const struct i2c_device_id *id) > > +{ > > + struct wacom_i2c *wac_i2c; > > + int err; > > + > > + if (!i2c_check_functionality(client->adapter, I2C_FUNC_I2C)) { > > + err = -EIO; > > + goto fail1; > > + } > > + > > + wac_i2c = kzalloc(sizeof(struct wacom_i2c), GFP_KERNEL); > > + wac_i2c->input = input_allocate_device(); > > + if (wac_i2c == NULL || wac_i2c->input == NULL) { > > + err = -ENOMEM; > > + goto fail2; > > + } > > + > > + wac_i2c->client = client; > > + i2c_set_clientdata(client, wac_i2c); > > + > > + mutex_init(&wac_i2c->lock); > > + > > + wac_i2c->input->name = "WACOM_I2C_DIGITIZER"; > > + wac_i2c->input->id.bustype = BUS_I2C; > > + wac_i2c->input->id.vendor = 0x56a; > > + wac_i2c->input->dev.parent = &client->dev; > > + wac_i2c->input->open = wacom_i2c_input_open; > > + wac_i2c->input->close = wacom_i2c_input_close; > > + wac_i2c->input->evbit[0] |= BIT_MASK(EV_KEY) | BIT_MASK(EV_ABS); > > + input_set_drvdata(wac_i2c->input, wac_i2c); > > + > > + if (input_register_device(wac_i2c->input)) { > > + dev_printk(KERN_ERR, &wac_i2c->client->dev, "Failed to > > register input device \n"); > > + err = -ENODEV; > > + goto fail3; > > + } > > + > > + return 0; > > + > > + fail3: > > + input_free_device(wac_i2c->input); > > + fail2: > > + kfree(wac_i2c); > > + dev_printk(KERN_ERR, &wac_i2c->client->dev, "Freeing device \n"); > > + fail1: > > + printk(KERN_ERR "wacom_i2c: No I2C functionality found \n"); > > + return err; > > +} > > + > > +static int wacom_i2c_remove(struct i2c_client *client) { > > + struct wacom_i2c *wac_i2c = i2c_get_clientdata(client); > > + > > + free_irq(EMR_IRQ, wac_i2c); > > + input_unregister_device(wac_i2c->input); > > + kfree(wac_i2c); > > + > > + return 0; > > +} > > + > > +#ifdef CONFIG_PM > > +static int wacom_i2c_suspend(struct device *dev) { > > + disable_irq(EMR_IRQ); > > + > > + return 0; > > +} > > + > > +static int wacom_i2c_resume(struct device *dev) { > > + enable_irq(EMR_IRQ); > > + > > + return 0; > > +} > > + > > +static const struct dev_pm_ops wacom_i2c_pm = { > > + .suspend = wacom_i2c_suspend, > > + .resume = wacom_i2c_resume, > > +}; > > +#endif > > + > > +static const struct i2c_device_id wacom_i2c_id[] = { > > + {WACNAME, 0}, > > + {}, > > +}; > > +MODULE_DEVICE_TABLE(i2c, wacom_i2c_id); > > + > > +static struct i2c_driver wacom_i2c_driver = { > > + .driver = { > > + .name = "wacom_i2c", > > + .owner = THIS_MODULE, > > +#ifdef CONFIG_PM > > + .pm = &wacom_i2c_pm, > > +#endif > > + }, > > + > > + .probe = wacom_i2c_probe, > > + .remove = wacom_i2c_remove, > > + .id_table = wacom_i2c_id, > > +}; > > + > > +static int __init wacom_i2c_init(void) > > +{ > > + return i2c_add_driver(&wacom_i2c_driver); > > +} > > + > > +static void __exit wacom_i2c_exit(void) { > > + i2c_del_driver(&wacom_i2c_driver); > > +} > > + > > +module_init(wacom_i2c_init); > > +module_exit(wacom_i2c_exit); > > + > > +MODULE_AUTHOR("Tatsunosuke Tobita <tobita.tatsunosuke@xxxxxxxxxxx>"); > > +MODULE_DESCRIPTION("WACOM EMR I2C Driver"); MODULE_LICENSE("GPL"); > > diff --git a/drivers/input/touchscreen/wacom_i2c.h > > b/drivers/input/touchscreen/wacom_i2c.h > > new file mode 100644 > > index 0000000..13d86f9 > > --- /dev/null > > +++ b/drivers/input/touchscreen/wacom_i2c.h > > @@ -0,0 +1,72 @@ > > +/* > > + * Wacom Penabled Driver for I2C > > + * > > + * Copyright (c) 2011 Tatsunosuke Tobita, Wacom. > > + * <tobita.tatsunosuke@xxxxxxxxxxx> > > + * > > + * 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 of 2 of the License, > > + * or (at your option) any later version. > > + */ > > + > > +#ifndef WACOM_I2C_H > > +#define WACOM_I2C_H > > + > > +#include <linux/module.h> > > +#include <linux/init.h> > > +#include <linux/kernel.h> > > +#include <linux/input.h> > > +#include <linux/i2c.h> > > +#include <linux/slab.h> > > +#include <linux/delay.h> > > +#include <linux/uaccess.h> > > +#include <linux/device.h> > > +#include <linux/irq.h> > > +#include <linux/interrupt.h> > > +#include <linux/gpio.h> > > +#include <asm/unaligned.h> > > +#include <asm/system.h> > > + > > +#define NAMEBUF 12 > > +#define WACNAME "WAC_I2C_EMR" > > + > > +/*Wacom Command*/ > > +#define COM_COORD_NUM 256 > > +#define COM_SAMPLERATE_40 0x33 > > +#define COM_SAMPLERATE_80 0x32 > > +#define COM_SAMPLERATE_133 0x31 > > +#define CMD_QUERY0 0x04 > > +#define CMD_QUERY1 0x00 > > +#define CMD_QUERY2 0x33 > > +#define CMD_QUERY3 0x02 > > +#define CMD_THROW0 0x05 > > +#define CMD_THROW1 0x00 > > +#define QUERY_SIZE 19 > > + > > +/*I2C address for digitizer*/ > > +#define WACOM_I2C_ADDR 0x09 > > + > > +/*Set a specific pin number on GPI for IRQ*/ #define GPIO 1 #define > > +EMR_IRQ gpio_to_irq(GPIO) > > + > > +/*Parameters for wacom own features*/ > > +struct wacom_features { > > + int x_max; > > + int y_max; > > + int pressure_max; > > + char fw_version; > > +}; > > + > > +/*Parameters for i2c driver*/ > > +struct wacom_i2c { > > + struct i2c_client *client; > > + struct input_dev *input; > > + struct mutex lock; > > + struct wacom_features wac_feature; > > + int type; > > + const char name[NAMEBUF]; > > +}; > > +#endif > > 1.7.1 > > > > -- > > To unsubscribe from this list: send the line "unsubscribe linux-input" in > > the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at > > http://vger.kernel.org/majordomo-info.html > 差出人: linux-input-owner@xxxxxxxxxxxxxxx は Tatsunosuke Tobita [junkpainting@xxxxxxxxx] の > 代理 > 送信日時: 2011年10月19日水曜日 15:36 > 宛先: linux-input@xxxxxxxxxxxxxxx > CC: Tatsunosuke Tobita > 件名: [PATCH] Input: Modified on Kconfig and Makefile for adding Wacom Stylus for I2C > > From: Tatsunosuke Tobita <tobita.tatsunosuke@xxxxxxxxxxx> > > > Signed-off-by: Tatsunosuke Tobita <tobita.tatsunosuke@xxxxxxxxxxx> > --- > drivers/input/touchscreen/Kconfig | 11 +++++++++++ > drivers/input/touchscreen/Makefile | 1 + > 2 files changed, 12 insertions(+), 0 deletions(-) > > diff --git a/drivers/input/touchscreen/Kconfig b/drivers/input/touchscreen/Kconfig > index 3488ffe..cae448e 100644 > --- a/drivers/input/touchscreen/Kconfig > +++ b/drivers/input/touchscreen/Kconfig > @@ -738,4 +738,15 @@ config TOUCHSCREEN_TPS6507X > To compile this driver as a module, choose M here: the > module will be called tps6507x_ts. > > +config TOUCHSCREEN_WACOM_I2C > + tristate "Wacom Tablet support(I2C)" > + depends on I2C > + help > + Say Y here if you want to use the I2C version of the Wacom Pen. > + Make sure to say Y to "Mouse support" (CONFIG_INPUT_MOUSEDEV) > + and/or "Event interface support" (CONFIG_INPUT_EVDEV) as well. > + > + To compile this driver as a module, choose M here: the module > + will be called wacom_i2c. > + > endif > diff --git a/drivers/input/touchscreen/Makefile b/drivers/input/touchscreen/Makefile > index f957676..61bbfc8 100644 > --- a/drivers/input/touchscreen/Makefile > +++ b/drivers/input/touchscreen/Makefile > @@ -61,3 +61,4 @@ obj-$(CONFIG_TOUCHSCREEN_WM97XX_MAINSTONE) += > mainstone-wm97xx.o > obj-$(CONFIG_TOUCHSCREEN_WM97XX_ZYLONITE) += zylonite-wm97xx.o > obj-$(CONFIG_TOUCHSCREEN_W90X900) += w90p910_ts.o > obj-$(CONFIG_TOUCHSCREEN_TPS6507X) += tps6507x-ts.o > +obj-$(CONFIG_TOUCHSCREEN_WACOM_I2C) += wacom_i2c.o > -- > 1.7.1 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-input" in the body of > a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at > http://vger.kernel.org/majordomo-info.html -- Dmitry -- To unsubscribe from this list: send the line "unsubscribe linux-input" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html