2342 lines
59 KiB
C
2342 lines
59 KiB
C
/*
|
|
* Atmel maXTouch Touchscreen Controller Driver
|
|
*
|
|
*
|
|
* Copyright (C) 2010 Atmel Corporation
|
|
* Copyright (C) 2010 Ulf Samuelsson (ulf@atmel.com)
|
|
* Copyright (C) 2009 Raphael Derosso Pereira <raphaelpereira@gmail.com>
|
|
*
|
|
* 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; version 2 of the License
|
|
*
|
|
* 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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*/
|
|
|
|
/*
|
|
*
|
|
* Driver for Atmel maXTouch family of touch controllers.
|
|
*
|
|
*/
|
|
|
|
#include <linux/i2c.h>
|
|
#include <linux/interrupt.h>
|
|
#include <linux/input.h>
|
|
#include <linux/debugfs.h>
|
|
#include <linux/cdev.h>
|
|
#include <linux/mutex.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/pm_runtime.h>
|
|
#include <linux/module.h>
|
|
|
|
#include <asm/uaccess.h>
|
|
|
|
#include <linux/atmel_maxtouch.h>
|
|
|
|
#if defined(CONFIG_HAS_EARLYSUSPEND)
|
|
#include <linux/earlysuspend.h>
|
|
|
|
/* Early-suspend level */
|
|
#define MXT_SUSPEND_LEVEL 1
|
|
#endif
|
|
|
|
|
|
#define DRIVER_VERSION "0.91a_mod"
|
|
|
|
static int debug = DEBUG_INFO;
|
|
static int comms = 0;
|
|
module_param(debug, int, 0644);
|
|
module_param(comms, int, 0644);
|
|
|
|
MODULE_PARM_DESC(debug, "Activate debugging output");
|
|
MODULE_PARM_DESC(comms, "Select communications mode");
|
|
|
|
#define T7_DATA_SIZE 3
|
|
|
|
/* Device Info descriptor */
|
|
/* Parsed from maXTouch "Id information" inside device */
|
|
struct mxt_device_info {
|
|
u8 family_id;
|
|
u8 variant_id;
|
|
u8 major;
|
|
u8 minor;
|
|
u8 build;
|
|
u8 num_objs;
|
|
u8 x_size;
|
|
u8 y_size;
|
|
char family_name[16]; /* Family name */
|
|
char variant_name[16]; /* Variant name */
|
|
u16 num_nodes; /* Number of sensor nodes */
|
|
};
|
|
|
|
/* object descriptor table, parsed from maXTouch "object table" */
|
|
struct mxt_object {
|
|
u16 chip_addr;
|
|
u8 type;
|
|
u8 size;
|
|
u8 instances;
|
|
u8 num_report_ids;
|
|
};
|
|
|
|
|
|
/* Mapping from report id to object type and instance */
|
|
struct report_id_map {
|
|
u8 object;
|
|
u8 instance;
|
|
/*
|
|
* This is the first report ID belonging to object. It enables us to
|
|
* find out easily the touch number: each touch has different report
|
|
* ID (which are assigned to touches in increasing order). By
|
|
* subtracting the first report ID from current, we get the touch
|
|
* number.
|
|
*/
|
|
u8 first_rid;
|
|
};
|
|
|
|
|
|
/* Driver datastructure */
|
|
struct mxt_data {
|
|
struct i2c_client *client;
|
|
struct input_dev *input;
|
|
char phys_name[32];
|
|
int irq;
|
|
|
|
u16 last_read_addr;
|
|
bool new_msgs;
|
|
u8 *last_message;
|
|
|
|
int valid_irq_counter;
|
|
int invalid_irq_counter;
|
|
int irq_counter;
|
|
int message_counter;
|
|
int read_fail_counter;
|
|
|
|
|
|
int bytes_to_read;
|
|
|
|
struct delayed_work dwork;
|
|
u8 xpos_format;
|
|
u8 ypos_format;
|
|
|
|
u8 numtouch;
|
|
|
|
struct mxt_device_info device_info;
|
|
|
|
u32 info_block_crc;
|
|
u32 configuration_crc;
|
|
u16 report_id_count;
|
|
struct report_id_map *rid_map;
|
|
struct mxt_object *object_table;
|
|
|
|
u16 msg_proc_addr;
|
|
u8 message_size;
|
|
|
|
u16 min_x_val;
|
|
u16 min_y_val;
|
|
u16 max_x_val;
|
|
u16 max_y_val;
|
|
|
|
int (*init_hw)(struct i2c_client *client);
|
|
int (*exit_hw)(struct i2c_client *client);
|
|
int (*power_on)(bool on);
|
|
u8 (*valid_interrupt)(void);
|
|
u8 (*read_chg)(void);
|
|
|
|
/* debugfs variables */
|
|
struct dentry *debug_dir;
|
|
int current_debug_datap;
|
|
|
|
struct mutex debug_mutex;
|
|
u16 *debug_data;
|
|
|
|
/* Character device variables */
|
|
struct cdev cdev;
|
|
struct cdev cdev_messages; /* 2nd Char dev for messages */
|
|
dev_t dev_num;
|
|
struct class *mxt_class;
|
|
|
|
|
|
u16 address_pointer;
|
|
bool valid_ap;
|
|
|
|
/* Message buffer & pointers */
|
|
char *messages;
|
|
int msg_buffer_startp, msg_buffer_endp;
|
|
/* Put only non-touch messages to buffer if this is set */
|
|
char nontouch_msg_only;
|
|
struct mutex msg_mutex;
|
|
#if defined(CONFIG_HAS_EARLYSUSPEND)
|
|
struct early_suspend early_suspend;
|
|
#endif
|
|
u8 t7_data[T7_DATA_SIZE];
|
|
bool is_suspended;
|
|
};
|
|
/*default value, enough to read versioning*/
|
|
#define CONFIG_DATA_SIZE 6
|
|
static u16 t38_size = CONFIG_DATA_SIZE;
|
|
static int mxt_read_block(struct i2c_client *client, u16 addr, u16 length,
|
|
u8 *value);
|
|
static int mxt_write_byte(struct i2c_client *client, u16 addr, u8 value);
|
|
static int mxt_write_block(struct i2c_client *client, u16 addr, u16 length,
|
|
u8 *value);
|
|
static u8 mxt_valid_interrupt_dummy(void)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
#define I2C_RETRY_COUNT 5
|
|
#define I2C_PAYLOAD_SIZE 254
|
|
|
|
/* Returns the start address of object in mXT memory. */
|
|
#define MXT_BASE_ADDR(object_type, mxt) \
|
|
get_object_address(object_type, 0, mxt->object_table, \
|
|
mxt->device_info.num_objs)
|
|
|
|
/* Maps a report ID to an object type (object type number). */
|
|
#define REPORT_ID_TO_OBJECT(rid, mxt) \
|
|
(((rid) == 0xff) ? 0 : mxt->rid_map[rid].object)
|
|
|
|
/* Maps a report ID to an object type (string). */
|
|
#define REPORT_ID_TO_OBJECT_NAME(rid, mxt) \
|
|
object_type_name[REPORT_ID_TO_OBJECT(rid, mxt)]
|
|
|
|
/* Returns non-zero if given object is a touch object */
|
|
#define IS_TOUCH_OBJECT(object) \
|
|
((object == MXT_TOUCH_MULTITOUCHSCREEN_T9) || \
|
|
(object == MXT_TOUCH_KEYARRAY_T15) || \
|
|
(object == MXT_TOUCH_PROXIMITY_T23) || \
|
|
(object == MXT_TOUCH_SINGLETOUCHSCREEN_T10) || \
|
|
(object == MXT_TOUCH_XSLIDER_T11) || \
|
|
(object == MXT_TOUCH_YSLIDER_T12) || \
|
|
(object == MXT_TOUCH_XWHEEL_T13) || \
|
|
(object == MXT_TOUCH_YWHEEL_T14) || \
|
|
(object == MXT_TOUCH_KEYSET_T31) || \
|
|
(object == MXT_TOUCH_XSLIDERSET_T32) ? 1 : 0)
|
|
|
|
#define mxt_debug(level, ...) \
|
|
do { \
|
|
if (debug >= (level)) \
|
|
pr_debug(__VA_ARGS__); \
|
|
} while (0)
|
|
|
|
|
|
/*
|
|
* Check whether we have multi-touch enabled kernel; if not, report just the
|
|
* first touch (on mXT224, the maximum is 10 simultaneous touches).
|
|
* Because just the 1st one is reported, it might seem that the screen is not
|
|
* responding to touch if the first touch is removed while the screen is being
|
|
* touched by another finger, so beware.
|
|
*
|
|
*/
|
|
|
|
#ifdef ABS_MT_TRACKING_ID
|
|
static inline void report_mt(int touch_number, int size, int x, int y, struct
|
|
mxt_data *mxt) {
|
|
input_report_abs(mxt->input, ABS_MT_TRACKING_ID, touch_number);
|
|
input_report_abs(mxt->input, ABS_MT_TOUCH_MAJOR, size);
|
|
input_report_abs(mxt->input, ABS_MT_POSITION_X, x);
|
|
input_report_abs(mxt->input, ABS_MT_POSITION_Y, y);
|
|
input_mt_sync(mxt->input);
|
|
}
|
|
#else
|
|
static inline void report_mt(int touch_number, int size, int x, int y, struct
|
|
mxt_data *mxt) {
|
|
if (touch_number == 0) {
|
|
input_report_abs(mxt->input, ABS_TOOL_WIDTH, size);
|
|
input_report_abs(mxt->input, ABS_X, x);
|
|
input_report_abs(mxt->input, ABS_Y, y);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
static inline void report_gesture(int data, struct mxt_data *mxt)
|
|
{
|
|
input_event(mxt->input, EV_MSC, MSC_GESTURE, data);
|
|
}
|
|
|
|
|
|
static const u8 *object_type_name[] = {
|
|
[0] = "Reserved",
|
|
[5] = "GEN_MESSAGEPROCESSOR_T5",
|
|
[6] = "GEN_COMMANDPROCESSOR_T6",
|
|
[7] = "GEN_POWERCONFIG_T7",
|
|
[8] = "GEN_ACQUIRECONFIG_T8",
|
|
[9] = "TOUCH_MULTITOUCHSCREEN_T9",
|
|
[15] = "TOUCH_KEYARRAY_T15",
|
|
[17] = "SPT_COMMSCONFIG_T18",
|
|
[19] = "SPT_GPIOPWM_T19",
|
|
[20] = "PROCI_GRIPFACESUPPRESSION_T20",
|
|
[22] = "PROCG_NOISESUPPRESSION_T22",
|
|
[23] = "TOUCH_PROXIMITY_T23",
|
|
[24] = "PROCI_ONETOUCHGESTUREPROCESSOR_T24",
|
|
[25] = "SPT_SELFTEST_T25",
|
|
[27] = "PROCI_TWOTOUCHGESTUREPROCESSOR_T27",
|
|
[28] = "SPT_CTECONFIG_T28",
|
|
[37] = "DEBUG_DIAGNOSTICS_T37",
|
|
[38] = "SPT_USER_DATA_T38",
|
|
[40] = "PROCI_GRIPSUPPRESSION_T40",
|
|
[41] = "PROCI_PALMSUPPRESSION_T41",
|
|
[42] = "PROCI_FACESUPPRESSION_T42",
|
|
[43] = "SPT_DIGITIZER_T43",
|
|
[44] = "SPT_MESSAGECOUNT_T44",
|
|
};
|
|
|
|
|
|
static u16 get_object_address(uint8_t object_type,
|
|
uint8_t instance,
|
|
struct mxt_object *object_table,
|
|
int max_objs);
|
|
|
|
int mxt_write_ap(struct mxt_data *mxt, u16 ap);
|
|
|
|
static int mxt_read_block_wo_addr(struct i2c_client *client,
|
|
u16 length,
|
|
u8 *value);
|
|
|
|
ssize_t debug_data_read(struct mxt_data *mxt, char *buf, size_t count,
|
|
loff_t *ppos, u8 debug_command){
|
|
int i;
|
|
u16 *data;
|
|
u16 diagnostics_reg;
|
|
int offset = 0;
|
|
int size;
|
|
int read_size;
|
|
int error;
|
|
char *buf_start;
|
|
u16 debug_data_addr;
|
|
u16 page_address;
|
|
u8 page;
|
|
u8 debug_command_reg;
|
|
|
|
data = mxt->debug_data;
|
|
if (data == NULL)
|
|
return -EIO;
|
|
|
|
/* If first read after open, read all data to buffer. */
|
|
if (mxt->current_debug_datap == 0){
|
|
|
|
diagnostics_reg = MXT_BASE_ADDR(MXT_GEN_COMMANDPROCESSOR_T6,
|
|
mxt) +
|
|
MXT_ADR_T6_DIAGNOSTIC;
|
|
if (count > (mxt->device_info.num_nodes * 2))
|
|
count = mxt->device_info.num_nodes;
|
|
|
|
debug_data_addr = MXT_BASE_ADDR(MXT_DEBUG_DIAGNOSTIC_T37, mxt)+
|
|
MXT_ADR_T37_DATA;
|
|
page_address = MXT_BASE_ADDR(MXT_DEBUG_DIAGNOSTIC_T37, mxt) +
|
|
MXT_ADR_T37_PAGE;
|
|
error = mxt_read_block(mxt->client, page_address, 1, &page);
|
|
if (error < 0)
|
|
return error;
|
|
mxt_debug(DEBUG_TRACE, "debug data page = %d\n", page);
|
|
while (page != 0) {
|
|
error = mxt_write_byte(mxt->client,
|
|
diagnostics_reg,
|
|
MXT_CMD_T6_PAGE_DOWN);
|
|
if (error < 0)
|
|
return error;
|
|
/* Wait for command to be handled; when it has, the
|
|
register will be cleared. */
|
|
debug_command_reg = 1;
|
|
while (debug_command_reg != 0) {
|
|
error = mxt_read_block(mxt->client,
|
|
diagnostics_reg, 1,
|
|
&debug_command_reg);
|
|
if (error < 0)
|
|
return error;
|
|
mxt_debug(DEBUG_TRACE,
|
|
"Waiting for debug diag command "
|
|
"to propagate...\n");
|
|
|
|
}
|
|
error = mxt_read_block(mxt->client, page_address, 1,
|
|
&page);
|
|
if (error < 0)
|
|
return error;
|
|
mxt_debug(DEBUG_TRACE, "debug data page = %d\n", page);
|
|
}
|
|
|
|
/*
|
|
* Lock mutex to prevent writing some unwanted data to debug
|
|
* command register. User can still write through the char
|
|
* device interface though. TODO: fix?
|
|
*/
|
|
|
|
mutex_lock(&mxt->debug_mutex);
|
|
/* Configure Debug Diagnostics object to show deltas/refs */
|
|
error = mxt_write_byte(mxt->client, diagnostics_reg,
|
|
debug_command);
|
|
|
|
/* Wait for command to be handled; when it has, the
|
|
* register will be cleared. */
|
|
debug_command_reg = 1;
|
|
while (debug_command_reg != 0) {
|
|
error = mxt_read_block(mxt->client,
|
|
diagnostics_reg, 1,
|
|
&debug_command_reg);
|
|
if (error < 0)
|
|
return error;
|
|
mxt_debug(DEBUG_TRACE, "Waiting for debug diag command "
|
|
"to propagate...\n");
|
|
|
|
}
|
|
|
|
if (error < 0) {
|
|
printk (KERN_WARNING
|
|
"Error writing to maXTouch device!\n");
|
|
return error;
|
|
}
|
|
|
|
size = mxt->device_info.num_nodes * sizeof(u16);
|
|
|
|
while (size > 0) {
|
|
read_size = size > 128 ? 128 : size;
|
|
mxt_debug(DEBUG_TRACE,
|
|
"Debug data read loop, reading %d bytes...\n",
|
|
read_size);
|
|
error = mxt_read_block(mxt->client,
|
|
debug_data_addr,
|
|
read_size,
|
|
(u8 *) &data[offset]);
|
|
if (error < 0) {
|
|
printk(KERN_WARNING
|
|
"Error reading debug data\n");
|
|
goto error;
|
|
}
|
|
offset += read_size/2;
|
|
size -= read_size;
|
|
|
|
/* Select next page */
|
|
error = mxt_write_byte(mxt->client, diagnostics_reg,
|
|
MXT_CMD_T6_PAGE_UP);
|
|
if (error < 0) {
|
|
printk(KERN_WARNING
|
|
"Error writing to maXTouch device!\n");
|
|
goto error;
|
|
}
|
|
}
|
|
mutex_unlock(&mxt->debug_mutex);
|
|
}
|
|
|
|
buf_start = buf;
|
|
i = mxt->current_debug_datap;
|
|
|
|
while (((buf- buf_start) < (count - 6)) &&
|
|
(i < mxt->device_info.num_nodes)){
|
|
|
|
mxt->current_debug_datap++;
|
|
if (debug_command == MXT_CMD_T6_REFERENCES_MODE)
|
|
buf += sprintf(buf, "%d: %5d\n", i,
|
|
(u16) le16_to_cpu(data[i]));
|
|
else if (debug_command == MXT_CMD_T6_DELTAS_MODE)
|
|
buf += sprintf(buf, "%d: %5d\n", i,
|
|
(s16) le16_to_cpu(data[i]));
|
|
i++;
|
|
}
|
|
|
|
return (buf - buf_start);
|
|
error:
|
|
mutex_unlock(&mxt->debug_mutex);
|
|
return error;
|
|
}
|
|
|
|
ssize_t deltas_read(struct file *file, char *buf, size_t count, loff_t *ppos)
|
|
{
|
|
return debug_data_read(file->private_data, buf, count, ppos,
|
|
MXT_CMD_T6_DELTAS_MODE);
|
|
}
|
|
|
|
ssize_t refs_read(struct file *file, char *buf, size_t count,
|
|
loff_t *ppos)
|
|
{
|
|
return debug_data_read(file->private_data, buf, count, ppos,
|
|
MXT_CMD_T6_REFERENCES_MODE);
|
|
}
|
|
|
|
int debug_data_open(struct inode *inode, struct file *file)
|
|
{
|
|
struct mxt_data *mxt;
|
|
int i;
|
|
mxt = inode->i_private;
|
|
if (mxt == NULL)
|
|
return -EIO;
|
|
mxt->current_debug_datap = 0;
|
|
mxt->debug_data = kmalloc(mxt->device_info.num_nodes * sizeof(u16),
|
|
GFP_KERNEL);
|
|
if (mxt->debug_data == NULL)
|
|
return -ENOMEM;
|
|
|
|
|
|
for (i = 0; i < mxt->device_info.num_nodes; i++)
|
|
mxt->debug_data[i] = 7777;
|
|
|
|
|
|
file->private_data = mxt;
|
|
return 0;
|
|
}
|
|
|
|
int debug_data_release(struct inode *inode, struct file *file)
|
|
{
|
|
struct mxt_data *mxt;
|
|
mxt = file->private_data;
|
|
kfree(mxt->debug_data);
|
|
return 0;
|
|
}
|
|
|
|
static struct file_operations delta_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = debug_data_open,
|
|
.release = debug_data_release,
|
|
.read = deltas_read,
|
|
};
|
|
|
|
static struct file_operations refs_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = debug_data_open,
|
|
.release = debug_data_release,
|
|
.read = refs_read,
|
|
};
|
|
|
|
|
|
int mxt_memory_open(struct inode *inode, struct file *file)
|
|
{
|
|
struct mxt_data *mxt;
|
|
mxt = container_of(inode->i_cdev, struct mxt_data, cdev);
|
|
if (mxt == NULL)
|
|
return -EIO;
|
|
file->private_data = mxt;
|
|
return 0;
|
|
}
|
|
|
|
int mxt_message_open(struct inode *inode, struct file *file)
|
|
{
|
|
struct mxt_data *mxt;
|
|
mxt = container_of(inode->i_cdev, struct mxt_data, cdev_messages);
|
|
if (mxt == NULL)
|
|
return -EIO;
|
|
file->private_data = mxt;
|
|
return 0;
|
|
}
|
|
|
|
|
|
ssize_t mxt_memory_read(struct file *file, char *buf, size_t count,
|
|
loff_t *ppos)
|
|
{
|
|
int i;
|
|
struct mxt_data *mxt;
|
|
|
|
mxt = file->private_data;
|
|
if (mxt->valid_ap){
|
|
mxt_debug(DEBUG_TRACE, "Reading %d bytes from current ap\n",
|
|
(int) count);
|
|
i = mxt_read_block_wo_addr(mxt->client, count, (u8 *) buf);
|
|
} else {
|
|
mxt_debug(DEBUG_TRACE, "Address pointer changed since set;"
|
|
"writing AP (%d) before reading %d bytes",
|
|
mxt->address_pointer, (int) count);
|
|
i = mxt_read_block(mxt->client, mxt->address_pointer, count,
|
|
buf);
|
|
}
|
|
|
|
return i;
|
|
}
|
|
|
|
ssize_t mxt_memory_write(struct file *file, const char *buf, size_t count,
|
|
loff_t *ppos)
|
|
{
|
|
int i;
|
|
int whole_blocks;
|
|
int last_block_size;
|
|
struct mxt_data *mxt;
|
|
u16 address;
|
|
|
|
mxt = file->private_data;
|
|
address = mxt->address_pointer;
|
|
|
|
mxt_debug(DEBUG_TRACE, "mxt_memory_write entered\n");
|
|
whole_blocks = count / I2C_PAYLOAD_SIZE;
|
|
last_block_size = count % I2C_PAYLOAD_SIZE;
|
|
|
|
for (i = 0; i < whole_blocks; i++) {
|
|
mxt_debug(DEBUG_TRACE, "About to write to %d...",
|
|
address);
|
|
mxt_write_block(mxt->client, address, I2C_PAYLOAD_SIZE,
|
|
(u8 *) buf);
|
|
address += I2C_PAYLOAD_SIZE;
|
|
buf += I2C_PAYLOAD_SIZE;
|
|
}
|
|
|
|
mxt_write_block(mxt->client, address, last_block_size, (u8 *) buf);
|
|
|
|
return count;
|
|
}
|
|
|
|
static long mxt_ioctl(struct file *file,
|
|
unsigned int cmd, unsigned long arg)
|
|
{
|
|
int retval;
|
|
struct mxt_data *mxt;
|
|
|
|
retval = 0;
|
|
mxt = file->private_data;
|
|
|
|
switch (cmd) {
|
|
case MXT_SET_ADDRESS_IOCTL:
|
|
retval = mxt_write_ap(mxt, (u16) arg);
|
|
if (retval >= 0) {
|
|
mxt->address_pointer = (u16) arg;
|
|
mxt->valid_ap = 1;
|
|
}
|
|
break;
|
|
case MXT_RESET_IOCTL:
|
|
retval = mxt_write_byte(mxt->client,
|
|
MXT_BASE_ADDR(MXT_GEN_COMMANDPROCESSOR_T6, mxt) +
|
|
MXT_ADR_T6_RESET,
|
|
1);
|
|
break;
|
|
case MXT_CALIBRATE_IOCTL:
|
|
retval = mxt_write_byte(mxt->client,
|
|
MXT_BASE_ADDR(MXT_GEN_COMMANDPROCESSOR_T6, mxt) +
|
|
MXT_ADR_T6_CALIBRATE,
|
|
1);
|
|
|
|
break;
|
|
case MXT_BACKUP_IOCTL:
|
|
retval = mxt_write_byte(mxt->client,
|
|
MXT_BASE_ADDR(MXT_GEN_COMMANDPROCESSOR_T6, mxt) +
|
|
MXT_ADR_T6_BACKUPNV,
|
|
MXT_CMD_T6_BACKUP);
|
|
break;
|
|
case MXT_NONTOUCH_MSG_IOCTL:
|
|
mxt->nontouch_msg_only = 1;
|
|
break;
|
|
case MXT_ALL_MSG_IOCTL:
|
|
mxt->nontouch_msg_only = 0;
|
|
break;
|
|
default:
|
|
return -EIO;
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Copies messages from buffer to user space.
|
|
*
|
|
* NOTE: if less than (mxt->message_size * 5 + 1) bytes requested,
|
|
* this will return 0!
|
|
*
|
|
*/
|
|
ssize_t mxt_message_read(struct file *file, char *buf, size_t count,
|
|
loff_t *ppos)
|
|
{
|
|
int i;
|
|
struct mxt_data *mxt;
|
|
char *buf_start;
|
|
|
|
mxt = file->private_data;
|
|
if (mxt == NULL)
|
|
return -EIO;
|
|
buf_start = buf;
|
|
|
|
mutex_lock(&mxt->msg_mutex);
|
|
/* Copy messages until buffer empty, or 'count' bytes written */
|
|
while ((mxt->msg_buffer_startp != mxt->msg_buffer_endp) &&
|
|
((buf - buf_start) < (count - (5 * mxt->message_size) - 1))){
|
|
|
|
for (i = 0; i < mxt->message_size; i++){
|
|
buf += sprintf(buf, "[%2X] ",
|
|
*(mxt->messages + mxt->msg_buffer_endp *
|
|
mxt->message_size + i));
|
|
}
|
|
buf += sprintf(buf, "\n");
|
|
if (mxt->msg_buffer_endp < MXT_MESSAGE_BUFFER_SIZE)
|
|
mxt->msg_buffer_endp++;
|
|
else
|
|
mxt->msg_buffer_endp = 0;
|
|
}
|
|
mutex_unlock(&mxt->msg_mutex);
|
|
return (buf - buf_start);
|
|
}
|
|
|
|
static struct file_operations mxt_message_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = mxt_message_open,
|
|
.read = mxt_message_read,
|
|
};
|
|
|
|
static struct file_operations mxt_memory_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = mxt_memory_open,
|
|
.read = mxt_memory_read,
|
|
.write = mxt_memory_write,
|
|
.unlocked_ioctl = mxt_ioctl,
|
|
};
|
|
|
|
|
|
/* Writes the address pointer (to set up following reads). */
|
|
|
|
int mxt_write_ap(struct mxt_data *mxt, u16 ap)
|
|
{
|
|
struct i2c_client *client;
|
|
__le16 le_ap = cpu_to_le16(ap);
|
|
client = mxt->client;
|
|
if (mxt != NULL)
|
|
mxt->last_read_addr = -1;
|
|
if (i2c_master_send(client, (u8 *) &le_ap, 2) == 2) {
|
|
mxt_debug(DEBUG_TRACE, "Address pointer set to %d\n", ap);
|
|
return 0;
|
|
} else {
|
|
mxt_debug(DEBUG_INFO, "Error writing address pointer!\n");
|
|
return -EIO;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
/* Calculates the 24-bit CRC sum. */
|
|
static u32 CRC_24(u32 crc, u8 byte1, u8 byte2)
|
|
{
|
|
static const u32 crcpoly = 0x80001B;
|
|
u32 result;
|
|
u32 data_word;
|
|
|
|
data_word = ((((u16) byte2) << 8u) | byte1);
|
|
result = ((crc << 1u) ^ data_word);
|
|
if (result & 0x1000000)
|
|
result ^= crcpoly;
|
|
return result;
|
|
}
|
|
|
|
/* Returns object address in mXT chip, or zero if object is not found */
|
|
static u16 get_object_address(uint8_t object_type,
|
|
uint8_t instance,
|
|
struct mxt_object *object_table,
|
|
int max_objs)
|
|
{
|
|
uint8_t object_table_index = 0;
|
|
uint8_t address_found = 0;
|
|
uint16_t address = 0;
|
|
struct mxt_object *obj;
|
|
|
|
while ((object_table_index < max_objs) && !address_found) {
|
|
obj = &object_table[object_table_index];
|
|
if (obj->type == object_type) {
|
|
address_found = 1;
|
|
/* Are there enough instances defined in the FW? */
|
|
if (obj->instances >= instance) {
|
|
address = obj->chip_addr +
|
|
(obj->size + 1) * instance;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
object_table_index++;
|
|
}
|
|
return address;
|
|
}
|
|
|
|
|
|
/*
|
|
* Reads a block of bytes from given address from mXT chip. If we are
|
|
* reading from message window, and previous read was from message window,
|
|
* there's no need to write the address pointer: the mXT chip will
|
|
* automatically set the address pointer back to message window start.
|
|
*/
|
|
|
|
static int mxt_read_block(struct i2c_client *client,
|
|
u16 addr,
|
|
u16 length,
|
|
u8 *value)
|
|
{
|
|
struct i2c_adapter *adapter = client->adapter;
|
|
struct i2c_msg msg[2];
|
|
__le16 le_addr;
|
|
struct mxt_data *mxt;
|
|
|
|
mxt = i2c_get_clientdata(client);
|
|
|
|
if (mxt != NULL) {
|
|
if ((mxt->last_read_addr == addr) &&
|
|
(addr == mxt->msg_proc_addr)) {
|
|
if (i2c_master_recv(client, value, length) == length)
|
|
return length;
|
|
else
|
|
return -EIO;
|
|
} else {
|
|
mxt->last_read_addr = addr;
|
|
}
|
|
}
|
|
|
|
mxt_debug(DEBUG_TRACE, "Writing address pointer & reading %d bytes "
|
|
"in on i2c transaction...\n", length);
|
|
|
|
le_addr = cpu_to_le16(addr);
|
|
msg[0].addr = client->addr;
|
|
msg[0].flags = 0x00;
|
|
msg[0].len = 2;
|
|
msg[0].buf = (u8 *) &le_addr;
|
|
|
|
msg[1].addr = client->addr;
|
|
msg[1].flags = I2C_M_RD;
|
|
msg[1].len = length;
|
|
msg[1].buf = (u8 *) value;
|
|
if (i2c_transfer(adapter, msg, 2) == 2)
|
|
return length;
|
|
else
|
|
return -EIO;
|
|
|
|
}
|
|
|
|
/* Reads a block of bytes from current address from mXT chip. */
|
|
|
|
static int mxt_read_block_wo_addr(struct i2c_client *client,
|
|
u16 length,
|
|
u8 *value)
|
|
{
|
|
|
|
|
|
if (i2c_master_recv(client, value, length) == length) {
|
|
mxt_debug(DEBUG_TRACE, "I2C block read ok\n");
|
|
return length;
|
|
} else {
|
|
mxt_debug(DEBUG_INFO, "I2C block read failed\n");
|
|
return -EIO;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
/* Writes one byte to given address in mXT chip. */
|
|
|
|
static int mxt_write_byte(struct i2c_client *client, u16 addr, u8 value)
|
|
{
|
|
struct {
|
|
__le16 le_addr;
|
|
u8 data;
|
|
|
|
} i2c_byte_transfer;
|
|
|
|
struct mxt_data *mxt;
|
|
|
|
mxt = i2c_get_clientdata(client);
|
|
if (mxt != NULL)
|
|
mxt->last_read_addr = -1;
|
|
i2c_byte_transfer.le_addr = cpu_to_le16(addr);
|
|
i2c_byte_transfer.data = value;
|
|
if (i2c_master_send(client, (u8 *) &i2c_byte_transfer, 3) == 3)
|
|
return 0;
|
|
else
|
|
return -EIO;
|
|
}
|
|
|
|
|
|
/* Writes a block of bytes (max 256) to given address in mXT chip. */
|
|
static int mxt_write_block(struct i2c_client *client,
|
|
u16 addr,
|
|
u16 length,
|
|
u8 *value)
|
|
{
|
|
int i;
|
|
struct {
|
|
__le16 le_addr;
|
|
u8 data[256];
|
|
|
|
} i2c_block_transfer;
|
|
|
|
struct mxt_data *mxt;
|
|
|
|
mxt_debug(DEBUG_TRACE, "Writing %d bytes to %d...", length, addr);
|
|
if (length > 256)
|
|
return -EINVAL;
|
|
mxt = i2c_get_clientdata(client);
|
|
if (mxt != NULL)
|
|
mxt->last_read_addr = -1;
|
|
for (i = 0; i < length; i++)
|
|
i2c_block_transfer.data[i] = *value++;
|
|
i2c_block_transfer.le_addr = cpu_to_le16(addr);
|
|
i = i2c_master_send(client, (u8 *) &i2c_block_transfer, length + 2);
|
|
if (i == (length + 2))
|
|
return length;
|
|
else
|
|
return -EIO;
|
|
}
|
|
|
|
/* Calculates the CRC value for mXT infoblock. */
|
|
int calculate_infoblock_crc(u32 *crc_result, u8 *data, int crc_area_size)
|
|
{
|
|
u32 crc = 0;
|
|
int i;
|
|
|
|
for (i = 0; i < (crc_area_size - 1); i = i + 2)
|
|
crc = CRC_24(crc, *(data + i), *(data + i + 1));
|
|
/* If uneven size, pad with zero */
|
|
if (crc_area_size & 0x0001)
|
|
crc = CRC_24(crc, *(data + i), 0);
|
|
/* Return only 24 bits of CRC. */
|
|
*crc_result = (crc & 0x00FFFFFF);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* Processes a touchscreen message. */
|
|
void process_T9_message(u8 *message, struct mxt_data *mxt, int last_touch)
|
|
{
|
|
|
|
struct input_dev *input;
|
|
u8 status;
|
|
u16 xpos = 0xFFFF;
|
|
u16 ypos = 0xFFFF;
|
|
u8 touch_size = 255;
|
|
u8 touch_number;
|
|
u8 amplitude;
|
|
u8 report_id;
|
|
|
|
static int stored_size[10];
|
|
static int stored_x[10];
|
|
static int stored_y[10];
|
|
int i;
|
|
int active_touches = 0;
|
|
/*
|
|
* If the 'last_touch' flag is set, we have received all the touch
|
|
* messages
|
|
* there are available in this cycle, so send the events for touches
|
|
* that are
|
|
* active.
|
|
*/
|
|
if (last_touch){
|
|
/* TODO: For compatibility with single-touch systems, send ABS_X &
|
|
* ABS_Y */
|
|
/*
|
|
if (stored_size[0]){
|
|
input_report_abs(mxt->input, ABS_X, stored_x[0]);
|
|
input_report_abs(mxt->input, ABS_Y, stored_y[0]);
|
|
}*/
|
|
|
|
|
|
for (i = 0; i < 10; i++){
|
|
if (stored_size[i]){
|
|
active_touches++;
|
|
input_report_abs(mxt->input,
|
|
ABS_MT_TRACKING_ID,
|
|
i);
|
|
input_report_abs(mxt->input,
|
|
ABS_MT_TOUCH_MAJOR,
|
|
stored_size[i]);
|
|
input_report_abs(mxt->input,
|
|
ABS_MT_POSITION_X,
|
|
stored_x[i]);
|
|
input_report_abs(mxt->input,
|
|
ABS_MT_POSITION_Y,
|
|
stored_y[i]);
|
|
input_mt_sync(mxt->input);
|
|
}
|
|
}
|
|
input_report_key(mxt->input, BTN_TOUCH, !!active_touches);
|
|
if (active_touches == 0)
|
|
input_mt_sync(mxt->input);
|
|
input_sync(mxt->input);
|
|
|
|
}else{
|
|
|
|
input = mxt->input;
|
|
status = message[MXT_MSG_T9_STATUS];
|
|
report_id = message[0];
|
|
|
|
if (status & MXT_MSGB_T9_SUPPRESS) {
|
|
/* Touch has been suppressed by grip/face */
|
|
/* detection */
|
|
mxt_debug(DEBUG_TRACE, "SUPRESS");
|
|
} else {
|
|
/* Put together the 10-/12-bit coordinate values. */
|
|
xpos = message[MXT_MSG_T9_XPOSMSB] * 16 +
|
|
((message[MXT_MSG_T9_XYPOSLSB] >> 4) & 0xF);
|
|
ypos = message[MXT_MSG_T9_YPOSMSB] * 16 +
|
|
((message[MXT_MSG_T9_XYPOSLSB] >> 0) & 0xF);
|
|
|
|
if (mxt->max_x_val < 1024)
|
|
xpos >>= 2;
|
|
if (mxt->max_y_val < 1024)
|
|
ypos >>= 2;
|
|
|
|
touch_number = message[MXT_MSG_REPORTID] -
|
|
mxt->rid_map[report_id].first_rid;
|
|
|
|
stored_x[touch_number] = xpos;
|
|
stored_y[touch_number] = ypos;
|
|
|
|
if (status & MXT_MSGB_T9_DETECT) {
|
|
/*
|
|
* mXT224 reports the number of touched nodes,
|
|
* so the exact value for touch ellipse major
|
|
* axis length in nodes would be 2*sqrt(touch_size/pi)
|
|
* (assuming round touch shape), which would then need
|
|
* to be scaled using information about how many sensor
|
|
* lines we do have. So this is very much simplified,
|
|
* but sufficient for most if not all apps?
|
|
*/
|
|
touch_size = message[MXT_MSG_T9_TCHAREA];
|
|
touch_size = touch_size >> 2;
|
|
if (!touch_size)
|
|
touch_size = 1;
|
|
/*
|
|
* report_mt(touch_number, touch_size, xpos, ypos, mxt);
|
|
*/
|
|
|
|
stored_size[touch_number] = touch_size;
|
|
|
|
if (status & MXT_MSGB_T9_AMP)
|
|
/* Amplitude of touch has changed */
|
|
amplitude = message[MXT_MSG_T9_TCHAMPLITUDE];
|
|
}
|
|
|
|
if (status & MXT_MSGB_T9_RELEASE) {
|
|
/* The previously reported touch has been removed.*/
|
|
/* report_mt(touch_number, 0, xpos, ypos, mxt); */
|
|
stored_size[touch_number] = 0;
|
|
}
|
|
|
|
/* input_sync(input); */
|
|
}
|
|
|
|
if (status & MXT_MSGB_T9_SUPPRESS) {
|
|
mxt_debug(DEBUG_TRACE, "SUPRESS");
|
|
} else {
|
|
if (status & MXT_MSGB_T9_DETECT) {
|
|
mxt_debug(DEBUG_TRACE, "DETECT:%s%s%s%s",
|
|
((status & MXT_MSGB_T9_PRESS) ? " PRESS" : ""),
|
|
((status & MXT_MSGB_T9_MOVE) ? " MOVE" : ""),
|
|
((status & MXT_MSGB_T9_AMP) ? " AMP" : ""),
|
|
((status & MXT_MSGB_T9_VECTOR) ? " VECT" : ""));
|
|
|
|
} else if (status & MXT_MSGB_T9_RELEASE) {
|
|
mxt_debug(DEBUG_TRACE, "RELEASE");
|
|
}
|
|
}
|
|
mxt_debug(DEBUG_TRACE, "X=%d, Y=%d, TOUCHSIZE=%d",
|
|
xpos, ypos, touch_size);
|
|
}
|
|
return;
|
|
}
|
|
|
|
|
|
int process_message(u8 *message, u8 object, struct mxt_data *mxt)
|
|
{
|
|
struct i2c_client *client;
|
|
u8 status;
|
|
u16 xpos = 0xFFFF;
|
|
u16 ypos = 0xFFFF;
|
|
u8 event;
|
|
u8 direction;
|
|
u16 distance;
|
|
u8 length;
|
|
u8 report_id;
|
|
static u8 error_cond = 0;
|
|
|
|
client = mxt->client;
|
|
length = mxt->message_size;
|
|
report_id = message[0];
|
|
|
|
if ((mxt->nontouch_msg_only == 0) ||
|
|
(!IS_TOUCH_OBJECT(object))){
|
|
mutex_lock(&mxt->msg_mutex);
|
|
/* Copy the message to buffer */
|
|
if (mxt->msg_buffer_startp < MXT_MESSAGE_BUFFER_SIZE) {
|
|
mxt->msg_buffer_startp++;
|
|
} else {
|
|
mxt->msg_buffer_startp = 0;
|
|
}
|
|
|
|
if (mxt->msg_buffer_startp == mxt->msg_buffer_endp) {
|
|
mxt_debug(DEBUG_TRACE,
|
|
"Message buf full, discarding last entry.\n");
|
|
if (mxt->msg_buffer_endp < MXT_MESSAGE_BUFFER_SIZE) {
|
|
mxt->msg_buffer_endp++;
|
|
} else {
|
|
mxt->msg_buffer_endp = 0;
|
|
}
|
|
}
|
|
memcpy((mxt->messages + mxt->msg_buffer_startp * length),
|
|
message,
|
|
length);
|
|
mutex_unlock(&mxt->msg_mutex);
|
|
}
|
|
|
|
switch (object) {
|
|
case MXT_GEN_COMMANDPROCESSOR_T6:
|
|
status = message[1];
|
|
|
|
if (status & MXT_MSGB_T6_COMSERR) {
|
|
if ((!error_cond) & MXT_MSGB_T6_COMSERR){
|
|
dev_err(&client->dev,
|
|
"maXTouch checksum error\n");
|
|
error_cond |= MXT_MSGB_T6_COMSERR;
|
|
}
|
|
}
|
|
if (status & MXT_MSGB_T6_CFGERR) {
|
|
/*
|
|
* Configuration error. A proper configuration
|
|
* needs to be written to chip and backed up.
|
|
*/
|
|
if ((!error_cond) & MXT_MSGB_T6_CFGERR){
|
|
dev_err(&client->dev,
|
|
"maXTouch configuration error\n");
|
|
error_cond |= MXT_MSGB_T6_CFGERR;
|
|
}
|
|
}
|
|
if (status & MXT_MSGB_T6_CAL) {
|
|
/* Calibration in action, no need to react */
|
|
dev_dbg(&client->dev,
|
|
"maXTouch calibration in progress\n");
|
|
}
|
|
if (status & MXT_MSGB_T6_SIGERR) {
|
|
/*
|
|
* Signal acquisition error, something is seriously
|
|
* wrong, not much we can in the driver to correct
|
|
* this
|
|
*/
|
|
if ((!error_cond) & MXT_MSGB_T6_SIGERR){
|
|
dev_err(&client->dev,
|
|
"maXTouch acquisition error\n");
|
|
error_cond |= MXT_MSGB_T6_SIGERR;
|
|
}
|
|
}
|
|
if (status & MXT_MSGB_T6_OFL) {
|
|
/*
|
|
* Cycle overflow, the acquisition interval is too
|
|
* short.
|
|
*/
|
|
dev_err(&client->dev,
|
|
"maXTouch cycle overflow\n");
|
|
}
|
|
if (status & MXT_MSGB_T6_RESET) {
|
|
/* Chip has reseted, no need to react. */
|
|
dev_dbg(&client->dev,
|
|
"maXTouch chip reset\n");
|
|
}
|
|
if (status == 0) {
|
|
/* Chip status back to normal. */
|
|
dev_dbg(&client->dev,
|
|
"maXTouch status normal\n");
|
|
error_cond = 0;
|
|
}
|
|
break;
|
|
|
|
case MXT_TOUCH_MULTITOUCHSCREEN_T9:
|
|
process_T9_message(message, mxt, 0);
|
|
break;
|
|
|
|
case MXT_SPT_GPIOPWM_T19:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving GPIO message\n");
|
|
break;
|
|
|
|
case MXT_PROCI_GRIPFACESUPPRESSION_T20:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving face suppression msg\n");
|
|
break;
|
|
|
|
case MXT_PROCG_NOISESUPPRESSION_T22:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving noise suppression msg\n");
|
|
status = message[MXT_MSG_T22_STATUS];
|
|
if (status & MXT_MSGB_T22_FHCHG) {
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"maXTouch: Freq changed\n");
|
|
}
|
|
if (status & MXT_MSGB_T22_GCAFERR) {
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"maXTouch: High noise "
|
|
"level\n");
|
|
}
|
|
if (status & MXT_MSGB_T22_FHERR) {
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"maXTouch: Freq changed - "
|
|
"Noise level too high\n");
|
|
}
|
|
break;
|
|
|
|
case MXT_PROCI_ONETOUCHGESTUREPROCESSOR_T24:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving one-touch gesture msg\n");
|
|
|
|
event = message[MXT_MSG_T24_STATUS] & 0x0F;
|
|
xpos = message[MXT_MSG_T24_XPOSMSB] * 16 +
|
|
((message[MXT_MSG_T24_XYPOSLSB] >> 4) & 0x0F);
|
|
ypos = message[MXT_MSG_T24_YPOSMSB] * 16 +
|
|
((message[MXT_MSG_T24_XYPOSLSB] >> 0) & 0x0F);
|
|
if (mxt->max_x_val < 1024)
|
|
xpos >>= 2;
|
|
if (mxt->max_y_val < 1024)
|
|
ypos >>= 2;
|
|
direction = message[MXT_MSG_T24_DIR];
|
|
distance = message[MXT_MSG_T24_DIST] +
|
|
(message[MXT_MSG_T24_DIST + 1] << 16);
|
|
|
|
report_gesture((event << 24) | (direction << 16) | distance,
|
|
mxt);
|
|
report_gesture((xpos << 16) | ypos, mxt);
|
|
|
|
break;
|
|
|
|
case MXT_SPT_SELFTEST_T25:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving Self-Test msg\n");
|
|
|
|
if (message[MXT_MSG_T25_STATUS] == MXT_MSGR_T25_OK) {
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"maXTouch: Self-Test OK\n");
|
|
|
|
} else {
|
|
dev_err(&client->dev,
|
|
"maXTouch: Self-Test Failed [%02x]:"
|
|
"{%02x,%02x,%02x,%02x,%02x}\n",
|
|
message[MXT_MSG_T25_STATUS],
|
|
message[MXT_MSG_T25_STATUS + 0],
|
|
message[MXT_MSG_T25_STATUS + 1],
|
|
message[MXT_MSG_T25_STATUS + 2],
|
|
message[MXT_MSG_T25_STATUS + 3],
|
|
message[MXT_MSG_T25_STATUS + 4]
|
|
);
|
|
}
|
|
break;
|
|
|
|
case MXT_PROCI_TWOTOUCHGESTUREPROCESSOR_T27:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving 2-touch gesture message\n");
|
|
|
|
event = message[MXT_MSG_T27_STATUS] & 0xF0;
|
|
xpos = message[MXT_MSG_T27_XPOSMSB] * 16 +
|
|
((message[MXT_MSG_T27_XYPOSLSB] >> 4) & 0x0F);
|
|
ypos = message[MXT_MSG_T27_YPOSMSB] * 16 +
|
|
((message[MXT_MSG_T27_XYPOSLSB] >> 0) & 0x0F);
|
|
if (mxt->max_x_val < 1024)
|
|
xpos >>= 2;
|
|
if (mxt->max_y_val < 1024)
|
|
ypos >>= 2;
|
|
|
|
direction = message[MXT_MSG_T27_ANGLE];
|
|
distance = message[MXT_MSG_T27_SEPARATION] +
|
|
(message[MXT_MSG_T27_SEPARATION + 1] << 16);
|
|
|
|
report_gesture((event << 24) | (direction << 16) | distance,
|
|
mxt);
|
|
report_gesture((xpos << 16) | ypos, mxt);
|
|
|
|
|
|
break;
|
|
|
|
case MXT_SPT_CTECONFIG_T28:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"Receiving CTE message...\n");
|
|
status = message[MXT_MSG_T28_STATUS];
|
|
if (status & MXT_MSGB_T28_CHKERR)
|
|
dev_err(&client->dev,
|
|
"maXTouch: Power-Up CRC failure\n");
|
|
|
|
break;
|
|
default:
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev,
|
|
"maXTouch: Unknown message!\n");
|
|
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
/*
|
|
* Processes messages when the interrupt line (CHG) is asserted. Keeps
|
|
* reading messages until a message with report ID 0xFF is received,
|
|
* which indicates that there is no more new messages.
|
|
*
|
|
*/
|
|
|
|
static void mxt_worker(struct work_struct *work)
|
|
{
|
|
struct mxt_data *mxt;
|
|
struct i2c_client *client;
|
|
|
|
u8 *message;
|
|
u16 message_length;
|
|
u16 message_addr;
|
|
u8 report_id;
|
|
u8 object;
|
|
int error;
|
|
int i;
|
|
char *message_string;
|
|
char *message_start;
|
|
|
|
message = NULL;
|
|
mxt = container_of(work, struct mxt_data, dwork.work);
|
|
client = mxt->client;
|
|
message_addr = mxt->msg_proc_addr;
|
|
message_length = mxt->message_size;
|
|
|
|
if (message_length < 256) {
|
|
message = kmalloc(message_length, GFP_KERNEL);
|
|
if (message == NULL) {
|
|
dev_err(&client->dev, "Error allocating memory\n");
|
|
goto fail_worker;
|
|
}
|
|
} else {
|
|
dev_err(&client->dev,
|
|
"Message length larger than 256 bytes not supported\n");
|
|
goto fail_worker;
|
|
}
|
|
|
|
mxt_debug(DEBUG_TRACE, "maXTouch worker active:\n");
|
|
|
|
do {
|
|
/* Read next message, reread on failure. */
|
|
/* TODO: message length, CRC included? */
|
|
mxt->message_counter++;
|
|
for (i = 1; i < I2C_RETRY_COUNT; i++) {
|
|
error = mxt_read_block(client,
|
|
message_addr,
|
|
message_length - 1,
|
|
message);
|
|
if (error >= 0)
|
|
break;
|
|
mxt->read_fail_counter++;
|
|
dev_err(&client->dev,
|
|
"Failure reading maxTouch device\n");
|
|
}
|
|
if (error < 0) {
|
|
kfree(message);
|
|
goto fail_worker;
|
|
}
|
|
|
|
if (mxt->address_pointer != message_addr)
|
|
mxt->valid_ap = 0;
|
|
report_id = message[0];
|
|
|
|
if (debug >= DEBUG_RAW) {
|
|
mxt_debug(DEBUG_RAW, "%s message [msg count: %08x]:",
|
|
REPORT_ID_TO_OBJECT_NAME(report_id, mxt),
|
|
mxt->message_counter
|
|
);
|
|
/* 5 characters per one byte */
|
|
message_string = kmalloc(message_length * 5,
|
|
GFP_KERNEL);
|
|
if (message_string == NULL) {
|
|
dev_err(&client->dev,
|
|
"Error allocating memory\n");
|
|
kfree(message);
|
|
goto fail_worker;
|
|
}
|
|
message_start = message_string;
|
|
for (i = 0; i < message_length; i++) {
|
|
message_string +=
|
|
sprintf(message_string,
|
|
"0x%02X ", message[i]);
|
|
}
|
|
mxt_debug(DEBUG_RAW, "%s", message_start);
|
|
kfree(message_start);
|
|
}
|
|
|
|
if ((report_id != MXT_END_OF_MESSAGES) && (report_id != 0)) {
|
|
memcpy(mxt->last_message, message, message_length);
|
|
mxt->new_msgs = 1;
|
|
smp_wmb();
|
|
/* Get type of object and process the message */
|
|
object = mxt->rid_map[report_id].object;
|
|
process_message(message, object, mxt);
|
|
}
|
|
|
|
mxt_debug(DEBUG_TRACE, "chgline: %d\n", mxt->read_chg());
|
|
} while (comms ? (mxt->read_chg() == 0) :
|
|
((report_id != MXT_END_OF_MESSAGES) && (report_id != 0)));
|
|
|
|
/* All messages processed, send the events) */
|
|
process_T9_message(NULL, mxt, 1);
|
|
|
|
kfree(message);
|
|
|
|
fail_worker:
|
|
/* Make sure we just didn't miss a interrupt. */
|
|
if (mxt->read_chg() == 0){
|
|
schedule_delayed_work(&mxt->dwork, 0);
|
|
} else
|
|
enable_irq(mxt->irq);
|
|
}
|
|
|
|
|
|
/*
|
|
* The maXTouch device will signal the host about a new message by asserting
|
|
* the CHG line. This ISR schedules a worker routine to read the message when
|
|
* that happens.
|
|
*/
|
|
|
|
static irqreturn_t mxt_irq_handler(int irq, void *_mxt)
|
|
{
|
|
struct mxt_data *mxt = _mxt;
|
|
|
|
mxt->irq_counter++;
|
|
if (mxt->valid_interrupt()) {
|
|
/* Send the signal only if falling edge generated the irq. */
|
|
disable_irq_nosync(mxt->irq);
|
|
schedule_delayed_work(&mxt->dwork, 0);
|
|
mxt->valid_irq_counter++;
|
|
} else {
|
|
mxt->invalid_irq_counter++;
|
|
return IRQ_NONE;
|
|
}
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
/* Initialization of driver */
|
|
/******************************************************************************/
|
|
|
|
static int __devinit mxt_identify(struct i2c_client *client,
|
|
struct mxt_data *mxt,
|
|
u8 *id_block_data)
|
|
{
|
|
u8 buf[MXT_ID_BLOCK_SIZE];
|
|
int error;
|
|
int identified;
|
|
|
|
identified = 0;
|
|
|
|
/* Read Device info to check if chip is valid */
|
|
error = mxt_read_block(client, MXT_ADDR_INFO_BLOCK, MXT_ID_BLOCK_SIZE,
|
|
(u8 *) buf);
|
|
|
|
if (error < 0) {
|
|
mxt->read_fail_counter++;
|
|
dev_err(&client->dev, "Failure accessing maXTouch device\n");
|
|
return -EIO;
|
|
}
|
|
|
|
memcpy(id_block_data, buf, MXT_ID_BLOCK_SIZE);
|
|
|
|
mxt->device_info.family_id = buf[0];
|
|
mxt->device_info.variant_id = buf[1];
|
|
mxt->device_info.major = ((buf[2] >> 4) & 0x0F);
|
|
mxt->device_info.minor = (buf[2] & 0x0F);
|
|
mxt->device_info.build = buf[3];
|
|
mxt->device_info.x_size = buf[4];
|
|
mxt->device_info.y_size = buf[5];
|
|
mxt->device_info.num_objs = buf[6];
|
|
mxt->device_info.num_nodes = mxt->device_info.x_size *
|
|
mxt->device_info.y_size;
|
|
|
|
/*
|
|
* Check Family & Variant Info; warn if not recognized but
|
|
* still continue.
|
|
*/
|
|
|
|
/* MXT224 */
|
|
if (mxt->device_info.family_id == MXT224_FAMILYID) {
|
|
strcpy(mxt->device_info.family_name, "mXT224");
|
|
|
|
if (mxt->device_info.variant_id == MXT224_CAL_VARIANTID) {
|
|
strcpy(mxt->device_info.variant_name, "Calibrated");
|
|
} else if (mxt->device_info.variant_id ==
|
|
MXT224_UNCAL_VARIANTID) {
|
|
strcpy(mxt->device_info.variant_name, "Uncalibrated");
|
|
} else {
|
|
dev_err(&client->dev,
|
|
"Warning: maXTouch Variant ID [%d] not "
|
|
"supported\n",
|
|
mxt->device_info.variant_id);
|
|
strcpy(mxt->device_info.variant_name, "UNKNOWN");
|
|
/* identified = -ENXIO; */
|
|
}
|
|
|
|
/* MXT1386 */
|
|
} else if (mxt->device_info.family_id == MXT1386_FAMILYID) {
|
|
strcpy(mxt->device_info.family_name, "mXT1386");
|
|
|
|
if (mxt->device_info.variant_id == MXT1386_CAL_VARIANTID) {
|
|
strcpy(mxt->device_info.variant_name, "Calibrated");
|
|
} else {
|
|
dev_err(&client->dev,
|
|
"Warning: maXTouch Variant ID [%d] not "
|
|
"supported\n",
|
|
mxt->device_info.variant_id);
|
|
strcpy(mxt->device_info.variant_name, "UNKNOWN");
|
|
/* identified = -ENXIO; */
|
|
}
|
|
/* Unknown family ID! */
|
|
} else {
|
|
dev_err(&client->dev,
|
|
"Warning: maXTouch Family ID [%d] not supported\n",
|
|
mxt->device_info.family_id);
|
|
strcpy(mxt->device_info.family_name, "UNKNOWN");
|
|
strcpy(mxt->device_info.variant_name, "UNKNOWN");
|
|
/* identified = -ENXIO; */
|
|
}
|
|
|
|
dev_info(
|
|
&client->dev,
|
|
"Atmel maXTouch (Family %s (%X), Variant %s (%X)) Firmware "
|
|
"version [%d.%d] Build %d\n",
|
|
mxt->device_info.family_name,
|
|
mxt->device_info.family_id,
|
|
mxt->device_info.variant_name,
|
|
mxt->device_info.variant_id,
|
|
mxt->device_info.major,
|
|
mxt->device_info.minor,
|
|
mxt->device_info.build
|
|
);
|
|
dev_dbg(
|
|
&client->dev,
|
|
"Atmel maXTouch Configuration "
|
|
"[X: %d] x [Y: %d]\n",
|
|
mxt->device_info.x_size,
|
|
mxt->device_info.y_size
|
|
);
|
|
return identified;
|
|
}
|
|
|
|
/*
|
|
* Reads the object table from maXTouch chip to get object data like
|
|
* address, size, report id. For Info Block CRC calculation, already read
|
|
* id data is passed to this function too (Info Block consists of the ID
|
|
* block and object table).
|
|
*
|
|
*/
|
|
static int __devinit mxt_read_object_table(struct i2c_client *client,
|
|
struct mxt_data *mxt,
|
|
u8 *raw_id_data)
|
|
{
|
|
u16 report_id_count;
|
|
u8 buf[MXT_OBJECT_TABLE_ELEMENT_SIZE];
|
|
u8 *raw_ib_data;
|
|
u8 object_type;
|
|
u16 object_address;
|
|
u16 object_size;
|
|
u8 object_instances;
|
|
u8 object_report_ids;
|
|
u16 object_info_address;
|
|
u32 crc;
|
|
u32 calculated_crc;
|
|
int i;
|
|
int error;
|
|
|
|
u8 object_instance;
|
|
u8 object_report_id;
|
|
u8 report_id;
|
|
int first_report_id;
|
|
int ib_pointer;
|
|
struct mxt_object *object_table;
|
|
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver reading configuration\n");
|
|
|
|
object_table = kzalloc(sizeof(struct mxt_object) *
|
|
mxt->device_info.num_objs,
|
|
GFP_KERNEL);
|
|
if (object_table == NULL) {
|
|
printk(KERN_WARNING "maXTouch: Memory allocation failed!\n");
|
|
error = -ENOMEM;
|
|
goto err_object_table_alloc;
|
|
}
|
|
|
|
raw_ib_data = kmalloc(MXT_OBJECT_TABLE_ELEMENT_SIZE *
|
|
mxt->device_info.num_objs + MXT_ID_BLOCK_SIZE,
|
|
GFP_KERNEL);
|
|
if (raw_ib_data == NULL) {
|
|
printk(KERN_WARNING "maXTouch: Memory allocation failed!\n");
|
|
error = -ENOMEM;
|
|
goto err_ib_alloc;
|
|
}
|
|
|
|
/* Copy the ID data for CRC calculation. */
|
|
memcpy(raw_ib_data, raw_id_data, MXT_ID_BLOCK_SIZE);
|
|
ib_pointer = MXT_ID_BLOCK_SIZE;
|
|
|
|
mxt->object_table = object_table;
|
|
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver Memory allocated\n");
|
|
|
|
object_info_address = MXT_ADDR_OBJECT_TABLE;
|
|
|
|
report_id_count = 0;
|
|
for (i = 0; i < mxt->device_info.num_objs; i++) {
|
|
mxt_debug(DEBUG_TRACE, "Reading maXTouch at [0x%04x]: ",
|
|
object_info_address);
|
|
|
|
error = mxt_read_block(client, object_info_address,
|
|
MXT_OBJECT_TABLE_ELEMENT_SIZE, buf);
|
|
|
|
if (error < 0) {
|
|
mxt->read_fail_counter++;
|
|
dev_err(&client->dev,
|
|
"maXTouch Object %d could not be read\n", i);
|
|
error = -EIO;
|
|
goto err_object_read;
|
|
}
|
|
|
|
memcpy(raw_ib_data + ib_pointer, buf,
|
|
MXT_OBJECT_TABLE_ELEMENT_SIZE);
|
|
ib_pointer += MXT_OBJECT_TABLE_ELEMENT_SIZE;
|
|
|
|
object_type = buf[0];
|
|
object_address = (buf[2] << 8) + buf[1];
|
|
object_size = buf[3] + 1;
|
|
object_instances = buf[4] + 1;
|
|
object_report_ids = buf[5];
|
|
mxt_debug(DEBUG_TRACE, "Type=%03d, Address=0x%04x, "
|
|
"Size=0x%02x, %d instances, %d report id's\n",
|
|
object_type,
|
|
object_address,
|
|
object_size,
|
|
object_instances,
|
|
object_report_ids
|
|
);
|
|
|
|
if (object_type == 38)
|
|
t38_size = object_size;
|
|
/* TODO: check whether object is known and supported? */
|
|
|
|
/* Save frequently needed info. */
|
|
if (object_type == MXT_GEN_MESSAGEPROCESSOR_T5) {
|
|
mxt->msg_proc_addr = object_address;
|
|
mxt->message_size = object_size;
|
|
}
|
|
|
|
object_table[i].type = object_type;
|
|
object_table[i].chip_addr = object_address;
|
|
object_table[i].size = object_size;
|
|
object_table[i].instances = object_instances;
|
|
object_table[i].num_report_ids = object_report_ids;
|
|
report_id_count += object_instances * object_report_ids;
|
|
|
|
object_info_address += MXT_OBJECT_TABLE_ELEMENT_SIZE;
|
|
}
|
|
|
|
mxt->rid_map =
|
|
kzalloc(sizeof(struct report_id_map) * (report_id_count + 1),
|
|
/* allocate for report_id 0, even if not used */
|
|
GFP_KERNEL);
|
|
if (mxt->rid_map == NULL) {
|
|
printk(KERN_WARNING "maXTouch: Can't allocate memory!\n");
|
|
error = -ENOMEM;
|
|
goto err_rid_map_alloc;
|
|
}
|
|
|
|
mxt->messages = kzalloc(mxt->message_size * MXT_MESSAGE_BUFFER_SIZE,
|
|
GFP_KERNEL);
|
|
if (mxt->messages == NULL) {
|
|
printk(KERN_WARNING "maXTouch: Can't allocate memory!\n");
|
|
error = -ENOMEM;
|
|
goto err_msg_alloc;
|
|
}
|
|
|
|
mxt->last_message = kzalloc(mxt->message_size, GFP_KERNEL);
|
|
if (mxt->last_message == NULL) {
|
|
printk(KERN_WARNING "maXTouch: Can't allocate memory!\n");
|
|
error = -ENOMEM;
|
|
goto err_msg_alloc;
|
|
}
|
|
|
|
mxt->report_id_count = report_id_count;
|
|
if (report_id_count > 254) { /* 0 & 255 are reserved */
|
|
dev_err(&client->dev,
|
|
"Too many maXTouch report id's [%d]\n",
|
|
report_id_count);
|
|
error = -ENXIO;
|
|
goto err_max_rid;
|
|
}
|
|
|
|
/* Create a mapping from report id to object type */
|
|
report_id = 1; /* Start from 1, 0 is reserved. */
|
|
|
|
/* Create table associating report id's with objects & instances */
|
|
for (i = 0; i < mxt->device_info.num_objs; i++) {
|
|
for (object_instance = 0;
|
|
object_instance < object_table[i].instances;
|
|
object_instance++){
|
|
first_report_id = report_id;
|
|
for (object_report_id = 0;
|
|
object_report_id < object_table[i].num_report_ids;
|
|
object_report_id++) {
|
|
mxt->rid_map[report_id].object =
|
|
object_table[i].type;
|
|
mxt->rid_map[report_id].instance =
|
|
object_instance;
|
|
mxt->rid_map[report_id].first_rid =
|
|
first_report_id;
|
|
report_id++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Read 3 byte CRC */
|
|
error = mxt_read_block(client, object_info_address, 3, buf);
|
|
if (error < 0) {
|
|
mxt->read_fail_counter++;
|
|
dev_err(&client->dev, "Error reading CRC\n");
|
|
}
|
|
|
|
crc = (buf[2] << 16) | (buf[1] << 8) | buf[0];
|
|
|
|
if (calculate_infoblock_crc(&calculated_crc, raw_ib_data,
|
|
ib_pointer)) {
|
|
printk(KERN_WARNING "Error while calculating CRC!\n");
|
|
calculated_crc = 0;
|
|
}
|
|
kfree(raw_ib_data);
|
|
|
|
mxt_debug(DEBUG_TRACE, "\nReported info block CRC = 0x%6X\n", crc);
|
|
mxt_debug(DEBUG_TRACE, "Calculated info block CRC = 0x%6X\n\n",
|
|
calculated_crc);
|
|
|
|
if (crc == calculated_crc) {
|
|
mxt->info_block_crc = crc;
|
|
} else {
|
|
mxt->info_block_crc = 0;
|
|
printk(KERN_ALERT "maXTouch: Info block CRC invalid!\n");
|
|
}
|
|
|
|
if (debug >= DEBUG_VERBOSE) {
|
|
|
|
dev_info(&client->dev, "maXTouch: %d Objects\n",
|
|
mxt->device_info.num_objs);
|
|
|
|
for (i = 0; i < mxt->device_info.num_objs; i++) {
|
|
dev_info(&client->dev, "Type:\t\t\t[%d]: %s\n",
|
|
object_table[i].type,
|
|
object_type_name[object_table[i].type]);
|
|
dev_info(&client->dev, "\tAddress:\t0x%04X\n",
|
|
object_table[i].chip_addr);
|
|
dev_info(&client->dev, "\tSize:\t\t%d Bytes\n",
|
|
object_table[i].size);
|
|
dev_info(&client->dev, "\tInstances:\t%d\n",
|
|
object_table[i].instances);
|
|
dev_info(&client->dev, "\tReport Id's:\t%d\n",
|
|
object_table[i].num_report_ids);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
|
|
|
|
err_max_rid:
|
|
kfree(mxt->last_message);
|
|
err_msg_alloc:
|
|
kfree(mxt->rid_map);
|
|
err_rid_map_alloc:
|
|
err_object_read:
|
|
kfree(raw_ib_data);
|
|
err_ib_alloc:
|
|
kfree(object_table);
|
|
err_object_table_alloc:
|
|
return error;
|
|
}
|
|
|
|
#if defined(CONFIG_PM)
|
|
static int mxt_suspend(struct device *dev)
|
|
{
|
|
struct mxt_data *mxt = dev_get_drvdata(dev);
|
|
int error, i;
|
|
u8 t7_deepsl_data[T7_DATA_SIZE];
|
|
u16 t7_addr;
|
|
|
|
if (device_may_wakeup(dev)) {
|
|
enable_irq_wake(mxt->irq);
|
|
return 0;
|
|
}
|
|
|
|
disable_irq(mxt->irq);
|
|
|
|
flush_delayed_work_sync(&mxt->dwork);
|
|
|
|
for (i = 0; i < T7_DATA_SIZE; i++)
|
|
t7_deepsl_data[i] = 0;
|
|
|
|
t7_addr = MXT_BASE_ADDR(MXT_GEN_POWERCONFIG_T7, mxt);
|
|
/* save current power state values */
|
|
error = mxt_read_block(mxt->client, t7_addr,
|
|
ARRAY_SIZE(mxt->t7_data), mxt->t7_data);
|
|
if (error < 0)
|
|
goto err_enable_irq;
|
|
|
|
/* configure deep sleep mode */
|
|
error = mxt_write_block(mxt->client, t7_addr,
|
|
ARRAY_SIZE(t7_deepsl_data), t7_deepsl_data);
|
|
if (error < 0)
|
|
goto err_enable_irq;
|
|
|
|
/* power off the device */
|
|
if (mxt->power_on) {
|
|
error = mxt->power_on(false);
|
|
if (error) {
|
|
dev_err(dev, "power off failed");
|
|
goto err_write_block;
|
|
}
|
|
}
|
|
mxt->is_suspended = true;
|
|
return 0;
|
|
|
|
err_write_block:
|
|
mxt_write_block(mxt->client, t7_addr,
|
|
ARRAY_SIZE(mxt->t7_data), mxt->t7_data);
|
|
err_enable_irq:
|
|
enable_irq(mxt->irq);
|
|
return error;
|
|
}
|
|
|
|
static int mxt_resume(struct device *dev)
|
|
{
|
|
struct mxt_data *mxt = dev_get_drvdata(dev);
|
|
int error;
|
|
u16 t7_addr;
|
|
|
|
if (device_may_wakeup(dev)) {
|
|
disable_irq_wake(mxt->irq);
|
|
return 0;
|
|
}
|
|
|
|
if (!mxt->is_suspended)
|
|
return 0;
|
|
|
|
/* power on the device */
|
|
if (mxt->power_on) {
|
|
error = mxt->power_on(true);
|
|
if (error) {
|
|
dev_err(dev, "power on failed");
|
|
return error;
|
|
}
|
|
}
|
|
|
|
t7_addr = MXT_BASE_ADDR(MXT_GEN_POWERCONFIG_T7, mxt);
|
|
/* restore the old power state values */
|
|
error = mxt_write_block(mxt->client, t7_addr,
|
|
ARRAY_SIZE(mxt->t7_data), mxt->t7_data);
|
|
if (error < 0)
|
|
goto err_write_block;
|
|
|
|
/* Make sure we just didn't miss a interrupt. */
|
|
if (mxt->read_chg() == 0)
|
|
schedule_delayed_work(&mxt->dwork, 0);
|
|
else
|
|
enable_irq(mxt->irq);
|
|
|
|
mxt->is_suspended = false;
|
|
|
|
return 0;
|
|
|
|
err_write_block:
|
|
if (mxt->power_on)
|
|
mxt->power_on(false);
|
|
return error;
|
|
}
|
|
|
|
#if defined(CONFIG_HAS_EARLYSUSPEND)
|
|
static void mxt_early_suspend(struct early_suspend *h)
|
|
{
|
|
struct mxt_data *mxt = container_of(h, struct mxt_data, early_suspend);
|
|
|
|
mxt_suspend(&mxt->client->dev);
|
|
}
|
|
|
|
static void mxt_late_resume(struct early_suspend *h)
|
|
{
|
|
struct mxt_data *mxt = container_of(h, struct mxt_data, early_suspend);
|
|
|
|
mxt_resume(&mxt->client->dev);
|
|
}
|
|
#endif
|
|
|
|
static const struct dev_pm_ops mxt_pm_ops = {
|
|
#ifndef CONFIG_HAS_EARLYSUSPEND
|
|
.suspend = mxt_suspend,
|
|
.resume = mxt_resume,
|
|
#endif
|
|
};
|
|
#endif
|
|
|
|
static int __devinit mxt_probe(struct i2c_client *client,
|
|
const struct i2c_device_id *id)
|
|
{
|
|
struct mxt_data *mxt;
|
|
struct maxtouch_platform_data *pdata;
|
|
struct input_dev *input;
|
|
u8 *id_data;
|
|
u8 *t38_data;
|
|
u16 t38_addr;
|
|
int error;
|
|
|
|
mxt_debug(DEBUG_INFO, "mXT224: mxt_probe\n");
|
|
|
|
if (client == NULL) {
|
|
pr_debug("maXTouch: client == NULL\n");
|
|
return -EINVAL;
|
|
} else if (client->adapter == NULL) {
|
|
pr_debug("maXTouch: client->adapter == NULL\n");
|
|
return -EINVAL;
|
|
} else if (&client->dev == NULL) {
|
|
pr_debug("maXTouch: client->dev == NULL\n");
|
|
return -EINVAL;
|
|
} else if (&client->adapter->dev == NULL) {
|
|
pr_debug("maXTouch: client->adapter->dev == NULL\n");
|
|
return -EINVAL;
|
|
} else if (id == NULL) {
|
|
pr_debug("maXTouch: id == NULL\n");
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* Enable runtime PM ops, start in ACTIVE mode */
|
|
error = pm_runtime_set_active(&client->dev);
|
|
if (error < 0)
|
|
dev_dbg(&client->dev, "unable to set runtime pm state\n");
|
|
pm_runtime_enable(&client->dev);
|
|
|
|
mxt_debug(DEBUG_INFO, "maXTouch driver v. %s\n", DRIVER_VERSION);
|
|
mxt_debug(DEBUG_INFO, "\t \"%s\"\n", client->name);
|
|
mxt_debug(DEBUG_INFO, "\taddr:\t0x%04x\n", client->addr);
|
|
mxt_debug(DEBUG_INFO, "\tirq:\t%d\n", client->irq);
|
|
mxt_debug(DEBUG_INFO, "\tflags:\t0x%04x\n", client->flags);
|
|
mxt_debug(DEBUG_INFO, "\tadapter:\"%s\"\n", client->adapter->name);
|
|
mxt_debug(DEBUG_INFO, "\tdevice:\t\"%s\"\n", client->dev.init_name);
|
|
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver functionality OK\n");
|
|
|
|
/* Allocate structure - we need it to identify device */
|
|
mxt = kzalloc(sizeof(struct mxt_data), GFP_KERNEL);
|
|
if (mxt == NULL) {
|
|
dev_err(&client->dev, "insufficient memory\n");
|
|
error = -ENOMEM;
|
|
goto err_mxt_alloc;
|
|
}
|
|
|
|
id_data = kmalloc(MXT_ID_BLOCK_SIZE, GFP_KERNEL);
|
|
if (id_data == NULL) {
|
|
dev_err(&client->dev, "insufficient memory\n");
|
|
error = -ENOMEM;
|
|
goto err_id_alloc;
|
|
}
|
|
|
|
input = input_allocate_device();
|
|
if (!input) {
|
|
dev_err(&client->dev, "error allocating input device\n");
|
|
error = -ENOMEM;
|
|
goto err_input_dev_alloc;
|
|
}
|
|
|
|
/* Initialize Platform data */
|
|
|
|
pdata = client->dev.platform_data;
|
|
if (pdata == NULL) {
|
|
dev_err(&client->dev, "platform data is required!\n");
|
|
error = -EINVAL;
|
|
goto err_pdata;
|
|
}
|
|
if (debug >= DEBUG_TRACE)
|
|
printk(KERN_INFO "Platform OK: pdata = 0x%08x\n",
|
|
(unsigned int) pdata);
|
|
|
|
mxt->is_suspended = false;
|
|
mxt->read_fail_counter = 0;
|
|
mxt->message_counter = 0;
|
|
|
|
if (pdata->min_x)
|
|
mxt->min_x_val = pdata->min_x;
|
|
else
|
|
mxt->min_x_val = 0;
|
|
|
|
if (pdata->min_y)
|
|
mxt->min_y_val = pdata->min_y;
|
|
else
|
|
mxt->min_y_val = 0;
|
|
|
|
mxt->max_x_val = pdata->max_x;
|
|
mxt->max_y_val = pdata->max_y;
|
|
|
|
/* Get data that is defined in board specific code. */
|
|
mxt->init_hw = pdata->init_platform_hw;
|
|
mxt->exit_hw = pdata->exit_platform_hw;
|
|
mxt->power_on = pdata->power_on;
|
|
mxt->read_chg = pdata->read_chg;
|
|
|
|
if (pdata->valid_interrupt != NULL)
|
|
mxt->valid_interrupt = pdata->valid_interrupt;
|
|
else
|
|
mxt->valid_interrupt = mxt_valid_interrupt_dummy;
|
|
|
|
if (mxt->init_hw) {
|
|
error = mxt->init_hw(client);
|
|
if (error) {
|
|
dev_err(&client->dev, "hw init failed");
|
|
goto err_init_hw;
|
|
}
|
|
}
|
|
|
|
/* power on the device */
|
|
if (mxt->power_on) {
|
|
error = mxt->power_on(true);
|
|
if (error) {
|
|
dev_err(&client->dev, "power on failed");
|
|
goto err_pwr_on;
|
|
}
|
|
}
|
|
|
|
if (debug >= DEBUG_TRACE)
|
|
printk(KERN_INFO "maXTouch driver identifying chip\n");
|
|
|
|
if (mxt_identify(client, mxt, id_data) < 0) {
|
|
dev_err(&client->dev, "Chip could not be identified\n");
|
|
error = -ENODEV;
|
|
goto err_identify;
|
|
}
|
|
/* Chip is valid and active. */
|
|
if (debug >= DEBUG_TRACE)
|
|
printk(KERN_INFO "maXTouch driver allocating input device\n");
|
|
|
|
mxt->client = client;
|
|
mxt->input = input;
|
|
|
|
INIT_DELAYED_WORK(&mxt->dwork, mxt_worker);
|
|
mutex_init(&mxt->debug_mutex);
|
|
mutex_init(&mxt->msg_mutex);
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver creating device name\n");
|
|
|
|
snprintf(
|
|
mxt->phys_name,
|
|
sizeof(mxt->phys_name),
|
|
"%s/input0",
|
|
dev_name(&client->dev)
|
|
);
|
|
input->name = "Atmel maXTouch Touchscreen controller";
|
|
input->phys = mxt->phys_name;
|
|
input->id.bustype = BUS_I2C;
|
|
input->dev.parent = &client->dev;
|
|
|
|
mxt_debug(DEBUG_INFO, "maXTouch name: \"%s\"\n", input->name);
|
|
mxt_debug(DEBUG_INFO, "maXTouch phys: \"%s\"\n", input->phys);
|
|
mxt_debug(DEBUG_INFO, "maXTouch driver setting abs parameters\n");
|
|
|
|
set_bit(BTN_TOUCH, input->keybit);
|
|
set_bit(INPUT_PROP_DIRECT, input->propbit);
|
|
|
|
/* Single touch */
|
|
input_set_abs_params(input, ABS_X, mxt->min_x_val,
|
|
mxt->max_x_val, 0, 0);
|
|
input_set_abs_params(input, ABS_Y, mxt->min_y_val,
|
|
mxt->max_y_val, 0, 0);
|
|
input_set_abs_params(input, ABS_PRESSURE, 0, MXT_MAX_REPORTED_PRESSURE,
|
|
0, 0);
|
|
input_set_abs_params(input, ABS_TOOL_WIDTH, 0, MXT_MAX_REPORTED_WIDTH,
|
|
0, 0);
|
|
|
|
/* Multitouch */
|
|
input_set_abs_params(input, ABS_MT_POSITION_X, mxt->min_x_val,
|
|
mxt->max_x_val, 0, 0);
|
|
input_set_abs_params(input, ABS_MT_POSITION_Y, mxt->min_y_val,
|
|
mxt->max_y_val, 0, 0);
|
|
input_set_abs_params(input, ABS_MT_TOUCH_MAJOR, 0, MXT_MAX_TOUCH_SIZE,
|
|
0, 0);
|
|
input_set_abs_params(input, ABS_MT_TRACKING_ID, 0, MXT_MAX_NUM_TOUCHES,
|
|
0, 0);
|
|
|
|
__set_bit(EV_ABS, input->evbit);
|
|
__set_bit(EV_SYN, input->evbit);
|
|
__set_bit(EV_KEY, input->evbit);
|
|
__set_bit(EV_MSC, input->evbit);
|
|
input->mscbit[0] = BIT_MASK(MSC_GESTURE);
|
|
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver setting client data\n");
|
|
i2c_set_clientdata(client, mxt);
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver setting drv data\n");
|
|
input_set_drvdata(input, mxt);
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver input register device\n");
|
|
error = input_register_device(mxt->input);
|
|
if (error < 0) {
|
|
dev_err(&client->dev,
|
|
"Failed to register input device\n");
|
|
goto err_register_device;
|
|
}
|
|
|
|
error = mxt_read_object_table(client, mxt, id_data);
|
|
if (error < 0)
|
|
goto err_read_ot;
|
|
|
|
|
|
/* Create debugfs entries. */
|
|
mxt->debug_dir = debugfs_create_dir("maXTouch", NULL);
|
|
if (mxt->debug_dir == ERR_PTR(-ENODEV)) {
|
|
/* debugfs is not enabled. */
|
|
printk(KERN_WARNING "debugfs not enabled in kernel\n");
|
|
} else if (mxt->debug_dir == NULL) {
|
|
printk(KERN_WARNING "error creating debugfs dir\n");
|
|
} else {
|
|
mxt_debug(DEBUG_TRACE, "created \"maXTouch\" debugfs dir\n");
|
|
|
|
debugfs_create_file("deltas", S_IRUSR, mxt->debug_dir, mxt,
|
|
&delta_fops);
|
|
debugfs_create_file("refs", S_IRUSR, mxt->debug_dir, mxt,
|
|
&refs_fops);
|
|
}
|
|
|
|
/* Create character device nodes for reading & writing registers */
|
|
mxt->mxt_class = class_create(THIS_MODULE, "maXTouch_memory");
|
|
if (IS_ERR(mxt->mxt_class)){
|
|
printk(KERN_WARNING "class create failed! exiting...");
|
|
goto err_class_create;
|
|
|
|
}
|
|
/* 2 numbers; one for memory and one for messages */
|
|
error = alloc_chrdev_region(&mxt->dev_num, 0, 2,
|
|
"maXTouch_memory");
|
|
mxt_debug(DEBUG_VERBOSE,
|
|
"device number %d allocated!\n", MAJOR(mxt->dev_num));
|
|
if (error){
|
|
printk(KERN_WARNING "Error registering device\n");
|
|
}
|
|
cdev_init(&mxt->cdev, &mxt_memory_fops);
|
|
cdev_init(&mxt->cdev_messages, &mxt_message_fops);
|
|
|
|
mxt_debug(DEBUG_VERBOSE, "cdev initialized\n");
|
|
mxt->cdev.owner = THIS_MODULE;
|
|
mxt->cdev_messages.owner = THIS_MODULE;
|
|
|
|
error = cdev_add(&mxt->cdev, mxt->dev_num, 1);
|
|
if (error){
|
|
printk(KERN_WARNING "Bad cdev\n");
|
|
}
|
|
|
|
error = cdev_add(&mxt->cdev_messages, mxt->dev_num + 1, 1);
|
|
if (error){
|
|
printk(KERN_WARNING "Bad cdev\n");
|
|
}
|
|
|
|
mxt_debug(DEBUG_VERBOSE, "cdev added\n");
|
|
|
|
device_create(mxt->mxt_class, NULL, MKDEV(MAJOR(mxt->dev_num), 0), NULL,
|
|
"maXTouch");
|
|
|
|
device_create(mxt->mxt_class, NULL, MKDEV(MAJOR(mxt->dev_num), 1), NULL,
|
|
"maXTouch_messages");
|
|
|
|
mxt->msg_buffer_startp = 0;
|
|
mxt->msg_buffer_endp = 0;
|
|
|
|
/* Allocate the interrupt */
|
|
mxt_debug(DEBUG_TRACE, "maXTouch driver allocating interrupt...\n");
|
|
mxt->irq = client->irq;
|
|
mxt->valid_irq_counter = 0;
|
|
mxt->invalid_irq_counter = 0;
|
|
mxt->irq_counter = 0;
|
|
if (mxt->irq) {
|
|
/* Try to request IRQ with falling edge first. This is
|
|
* not always supported. If it fails, try with any edge. */
|
|
error = request_irq(mxt->irq,
|
|
mxt_irq_handler,
|
|
IRQF_TRIGGER_FALLING,
|
|
client->dev.driver->name,
|
|
mxt);
|
|
if (error < 0) {
|
|
/* TODO: why only 0 works on STK1000? */
|
|
error = request_irq(mxt->irq,
|
|
mxt_irq_handler,
|
|
0,
|
|
client->dev.driver->name,
|
|
mxt);
|
|
}
|
|
|
|
if (error < 0) {
|
|
dev_err(&client->dev,
|
|
"failed to allocate irq %d\n", mxt->irq);
|
|
goto err_irq;
|
|
}
|
|
}
|
|
|
|
if (debug > DEBUG_INFO)
|
|
dev_info(&client->dev, "touchscreen, irq %d\n", mxt->irq);
|
|
|
|
t38_data = kmalloc(t38_size*sizeof(u8), GFP_KERNEL);
|
|
|
|
if (t38_data == NULL) {
|
|
dev_err(&client->dev, "insufficient memory\n");
|
|
error = -ENOMEM;
|
|
goto err_t38;
|
|
}
|
|
|
|
t38_addr = MXT_BASE_ADDR(MXT_USER_INFO_T38, mxt);
|
|
mxt_read_block(client, t38_addr, t38_size, t38_data);
|
|
dev_info(&client->dev, "VERSION:%02x.%02x.%02x, DATE: %d/%d/%d\n",
|
|
t38_data[0], t38_data[1], t38_data[2],
|
|
t38_data[3], t38_data[4], t38_data[5]);
|
|
|
|
/* Schedule a worker routine to read any messages that might have
|
|
* been sent before interrupts were enabled. */
|
|
cancel_delayed_work(&mxt->dwork);
|
|
disable_irq(mxt->irq);
|
|
schedule_delayed_work(&mxt->dwork, 0);
|
|
kfree(t38_data);
|
|
kfree(id_data);
|
|
|
|
device_init_wakeup(&client->dev, pdata->wakeup);
|
|
#if defined(CONFIG_HAS_EARLYSUSPEND)
|
|
mxt->early_suspend.level = EARLY_SUSPEND_LEVEL_BLANK_SCREEN +
|
|
MXT_SUSPEND_LEVEL;
|
|
mxt->early_suspend.suspend = mxt_early_suspend;
|
|
mxt->early_suspend.resume = mxt_late_resume;
|
|
register_early_suspend(&mxt->early_suspend);
|
|
#endif
|
|
|
|
return 0;
|
|
|
|
err_t38:
|
|
free_irq(mxt->irq, mxt);
|
|
err_irq:
|
|
kfree(mxt->rid_map);
|
|
kfree(mxt->object_table);
|
|
kfree(mxt->last_message);
|
|
err_class_create:
|
|
if (mxt->debug_dir)
|
|
debugfs_remove(mxt->debug_dir);
|
|
kfree(mxt->last_message);
|
|
kfree(mxt->rid_map);
|
|
kfree(mxt->object_table);
|
|
err_read_ot:
|
|
input_unregister_device(mxt->input);
|
|
mxt->input = NULL;
|
|
err_register_device:
|
|
mutex_destroy(&mxt->debug_mutex);
|
|
mutex_destroy(&mxt->msg_mutex);
|
|
err_identify:
|
|
if (mxt->power_on)
|
|
mxt->power_on(false);
|
|
err_pwr_on:
|
|
if (mxt->exit_hw != NULL)
|
|
mxt->exit_hw(client);
|
|
err_init_hw:
|
|
err_pdata:
|
|
input_free_device(input);
|
|
err_input_dev_alloc:
|
|
kfree(id_data);
|
|
err_id_alloc:
|
|
kfree(mxt);
|
|
err_mxt_alloc:
|
|
pm_runtime_set_suspended(&client->dev);
|
|
pm_runtime_disable(&client->dev);
|
|
return error;
|
|
}
|
|
|
|
static int __devexit mxt_remove(struct i2c_client *client)
|
|
{
|
|
struct mxt_data *mxt;
|
|
|
|
pm_runtime_set_suspended(&client->dev);
|
|
pm_runtime_disable(&client->dev);
|
|
|
|
mxt = i2c_get_clientdata(client);
|
|
|
|
/* Remove debug dir entries */
|
|
debugfs_remove_recursive(mxt->debug_dir);
|
|
|
|
device_init_wakeup(&client->dev, 0);
|
|
#if defined(CONFIG_HAS_EARLYSUSPEND)
|
|
unregister_early_suspend(&mxt->early_suspend);
|
|
#endif
|
|
|
|
if (mxt != NULL) {
|
|
if (mxt->power_on)
|
|
mxt->power_on(false);
|
|
|
|
if (mxt->exit_hw != NULL)
|
|
mxt->exit_hw(client);
|
|
|
|
if (mxt->irq) {
|
|
free_irq(mxt->irq, mxt);
|
|
}
|
|
|
|
unregister_chrdev_region(mxt->dev_num, 2);
|
|
device_destroy(mxt->mxt_class, MKDEV(MAJOR(mxt->dev_num), 0));
|
|
device_destroy(mxt->mxt_class, MKDEV(MAJOR(mxt->dev_num), 1));
|
|
cdev_del(&mxt->cdev);
|
|
cdev_del(&mxt->cdev_messages);
|
|
cancel_delayed_work_sync(&mxt->dwork);
|
|
input_unregister_device(mxt->input);
|
|
class_destroy(mxt->mxt_class);
|
|
debugfs_remove(mxt->debug_dir);
|
|
|
|
kfree(mxt->rid_map);
|
|
kfree(mxt->object_table);
|
|
kfree(mxt->last_message);
|
|
}
|
|
kfree(mxt);
|
|
|
|
i2c_set_clientdata(client, NULL);
|
|
if (debug >= DEBUG_TRACE)
|
|
dev_info(&client->dev, "Touchscreen unregistered\n");
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct i2c_device_id mxt_idtable[] = {
|
|
{"maXTouch", 0,},
|
|
{ }
|
|
};
|
|
|
|
MODULE_DEVICE_TABLE(i2c, mxt_idtable);
|
|
|
|
static struct i2c_driver mxt_driver = {
|
|
.driver = {
|
|
.name = "maXTouch",
|
|
.owner = THIS_MODULE,
|
|
#if defined(CONFIG_PM)
|
|
.pm = &mxt_pm_ops,
|
|
#endif
|
|
},
|
|
|
|
.id_table = mxt_idtable,
|
|
.probe = mxt_probe,
|
|
.remove = __devexit_p(mxt_remove),
|
|
};
|
|
|
|
static int __init mxt_init(void)
|
|
{
|
|
int err;
|
|
err = i2c_add_driver(&mxt_driver);
|
|
if (err) {
|
|
printk(KERN_WARNING "Adding maXTouch driver failed "
|
|
"(errno = %d)\n", err);
|
|
} else {
|
|
mxt_debug(DEBUG_TRACE, "Successfully added driver %s\n",
|
|
mxt_driver.driver.name);
|
|
}
|
|
return err;
|
|
}
|
|
|
|
static void __exit mxt_cleanup(void)
|
|
{
|
|
i2c_del_driver(&mxt_driver);
|
|
}
|
|
|
|
|
|
module_init(mxt_init);
|
|
module_exit(mxt_cleanup);
|
|
|
|
MODULE_AUTHOR("Iiro Valkonen");
|
|
MODULE_DESCRIPTION("Driver for Atmel maXTouch Touchscreen Controller");
|
|
MODULE_LICENSE("GPL");
|