| /* |
| * Synaptics DSX touchscreen driver |
| * |
| * Copyright (C) 2012 Synaptics Incorporated |
| * |
| * Copyright (C) 2012 Alexandra Chin <alexandra.chin@tw.synaptics.com> |
| * Copyright (C) 2012 Scott Lin <scott.lin@tw.synaptics.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; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| */ |
| #include <linux/kernel.h> |
| #include <linux/module.h> |
| #include <linux/slab.h> |
| #include <linux/interrupt.h> |
| #include <linux/delay.h> |
| #include <linux/input.h> |
| #include <linux/signal.h> |
| #include <linux/sched.h> |
| #include <linux/gpio.h> |
| #include <linux/uaccess.h> |
| #include <linux/cdev.h> |
| #include <linux/platform_device.h> |
| #include <linux/input/synaptics_dsx_new.h> |
| #include "synaptics_dsx_core.h" |
| |
| #define CHAR_DEVICE_NAME "rmi" |
| #define DEVICE_CLASS_NAME "rmidev" |
| #define SYSFS_FOLDER_NAME "rmidev" |
| #define DEV_NUMBER 1 |
| #define REG_ADDR_LIMIT 0xFFFF |
| |
| static ssize_t rmidev_sysfs_data_show(struct file *data_file, |
| struct kobject *kobj, struct bin_attribute *attributes, |
| char *buf, loff_t pos, size_t count); |
| |
| static ssize_t rmidev_sysfs_data_store(struct file *data_file, |
| struct kobject *kobj, struct bin_attribute *attributes, |
| char *buf, loff_t pos, size_t count); |
| |
| static ssize_t rmidev_sysfs_open_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count); |
| |
| static ssize_t rmidev_sysfs_release_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count); |
| |
| static ssize_t rmidev_sysfs_attn_state_show(struct device *dev, |
| struct device_attribute *attr, char *buf); |
| |
| static ssize_t rmidev_sysfs_pid_show(struct device *dev, |
| struct device_attribute *attr, char *buf); |
| |
| static ssize_t rmidev_sysfs_pid_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count); |
| |
| static ssize_t rmidev_sysfs_term_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count); |
| |
| static ssize_t rmidev_sysfs_intr_mask_show(struct device *dev, |
| struct device_attribute *attr, char *buf); |
| |
| static ssize_t rmidev_sysfs_intr_mask_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count); |
| |
| struct rmidev_handle { |
| dev_t dev_no; |
| pid_t pid; |
| unsigned char intr_mask; |
| struct device dev; |
| struct synaptics_rmi4_data *rmi4_data; |
| struct kobject *sysfs_dir; |
| struct siginfo interrupt_signal; |
| struct siginfo terminate_signal; |
| struct task_struct *task; |
| void *data; |
| bool irq_enabled; |
| }; |
| |
| struct rmidev_data { |
| int ref_count; |
| struct cdev main_dev; |
| struct class *device_class; |
| struct mutex file_mutex; |
| struct rmidev_handle *rmi_dev; |
| }; |
| |
| static struct bin_attribute attr_data = { |
| .attr = { |
| .name = "data", |
| .mode = (S_IRUGO | S_IWUGO), |
| }, |
| .size = 0, |
| .read = rmidev_sysfs_data_show, |
| .write = rmidev_sysfs_data_store, |
| }; |
| |
| static struct device_attribute attrs[] = { |
| __ATTR(open, S_IWUGO, |
| synaptics_rmi4_show_error, |
| rmidev_sysfs_open_store), |
| __ATTR(release, S_IWUGO, |
| synaptics_rmi4_show_error, |
| rmidev_sysfs_release_store), |
| __ATTR(attn_state, S_IRUGO, |
| rmidev_sysfs_attn_state_show, |
| synaptics_rmi4_store_error), |
| __ATTR(pid, S_IRUGO | S_IWUGO, |
| rmidev_sysfs_pid_show, |
| rmidev_sysfs_pid_store), |
| __ATTR(term, S_IWUGO, |
| synaptics_rmi4_show_error, |
| rmidev_sysfs_term_store), |
| __ATTR(intr_mask, S_IRUGO | S_IWUGO, |
| rmidev_sysfs_intr_mask_show, |
| rmidev_sysfs_intr_mask_store), |
| }; |
| |
| static int rmidev_major_num; |
| |
| static struct class *rmidev_device_class; |
| |
| static struct rmidev_handle *rmidev; |
| |
| DECLARE_COMPLETION(rmidev_remove_complete); |
| |
| static irqreturn_t rmidev_sysfs_irq(int irq, void *data) |
| { |
| struct synaptics_rmi4_data *rmi4_data = data; |
| |
| sysfs_notify(&rmi4_data->input_dev->dev.kobj, |
| SYSFS_FOLDER_NAME, "attn_state"); |
| |
| return IRQ_HANDLED; |
| } |
| |
| static int rmidev_sysfs_irq_enable(struct synaptics_rmi4_data *rmi4_data, |
| bool enable) |
| { |
| int retval = 0; |
| unsigned char intr_status[MAX_INTR_REGISTERS]; |
| unsigned long irq_flags = IRQF_TRIGGER_FALLING | IRQF_TRIGGER_RISING; |
| |
| if (enable) { |
| if (rmidev->irq_enabled) |
| return retval; |
| |
| /* Clear interrupts first */ |
| retval = synaptics_rmi4_reg_read(rmi4_data, |
| rmi4_data->f01_data_base_addr + 1, |
| intr_status, |
| rmi4_data->num_of_intr_regs); |
| if (retval < 0) |
| return retval; |
| |
| retval = request_threaded_irq(rmi4_data->irq, NULL, |
| rmidev_sysfs_irq, irq_flags, |
| PLATFORM_DRIVER_NAME, rmi4_data); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to create irq thread\n", |
| __func__); |
| return retval; |
| } |
| |
| rmidev->irq_enabled = true; |
| } else { |
| if (rmidev->irq_enabled) { |
| disable_irq(rmi4_data->irq); |
| free_irq(rmi4_data->irq, rmi4_data); |
| rmidev->irq_enabled = false; |
| } |
| } |
| |
| return retval; |
| } |
| |
| static ssize_t rmidev_sysfs_data_show(struct file *data_file, |
| struct kobject *kobj, struct bin_attribute *attributes, |
| char *buf, loff_t pos, size_t count) |
| { |
| int retval; |
| unsigned int length = (unsigned int)count; |
| unsigned short address = (unsigned short)pos; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (length > (REG_ADDR_LIMIT - address)) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Out of register map limit\n", |
| __func__); |
| return -EINVAL; |
| } |
| |
| if (length) { |
| retval = synaptics_rmi4_reg_read(rmi4_data, |
| address, |
| (unsigned char *)buf, |
| length); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to read data\n", |
| __func__); |
| return retval; |
| } |
| } else { |
| return -EINVAL; |
| } |
| |
| return length; |
| } |
| |
| static ssize_t rmidev_sysfs_data_store(struct file *data_file, |
| struct kobject *kobj, struct bin_attribute *attributes, |
| char *buf, loff_t pos, size_t count) |
| { |
| int retval; |
| unsigned int length = (unsigned int)count; |
| unsigned short address = (unsigned short)pos; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (length > (REG_ADDR_LIMIT - address)) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Out of register map limit\n", |
| __func__); |
| return -EINVAL; |
| } |
| |
| if (length) { |
| retval = synaptics_rmi4_reg_write(rmi4_data, |
| address, |
| (unsigned char *)buf, |
| length); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to write data\n", |
| __func__); |
| return retval; |
| } |
| } else { |
| return -EINVAL; |
| } |
| |
| return length; |
| } |
| |
| static ssize_t rmidev_sysfs_open_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count) |
| { |
| unsigned int input; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (sscanf(buf, "%u", &input) != 1) |
| return -EINVAL; |
| |
| if (input != 1) |
| return -EINVAL; |
| |
| rmi4_data->irq_enable(rmi4_data, false, false); |
| rmidev_sysfs_irq_enable(rmi4_data, true); |
| |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: Attention interrupt disabled\n", |
| __func__); |
| |
| return count; |
| } |
| |
| static ssize_t rmidev_sysfs_release_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count) |
| { |
| unsigned int input; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (sscanf(buf, "%u", &input) != 1) |
| return -EINVAL; |
| |
| if (input != 1) |
| return -EINVAL; |
| |
| rmidev_sysfs_irq_enable(rmi4_data, false); |
| rmi4_data->irq_enable(rmi4_data, true, false); |
| |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: Attention interrupt enabled\n", |
| __func__); |
| |
| rmi4_data->reset_device(rmi4_data); |
| |
| return count; |
| } |
| |
| static ssize_t rmidev_sysfs_attn_state_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| int attn_state; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| const struct synaptics_dsx_board_data *bdata = |
| rmi4_data->hw_if->board_data; |
| |
| attn_state = gpio_get_value(bdata->irq_gpio); |
| |
| return snprintf(buf, PAGE_SIZE, "%u\n", attn_state); |
| } |
| |
| static ssize_t rmidev_sysfs_pid_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| return snprintf(buf, PAGE_SIZE, "%u\n", rmidev->pid); |
| } |
| |
| static ssize_t rmidev_sysfs_pid_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count) |
| { |
| unsigned int input; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (sscanf(buf, "%u", &input) != 1) |
| return -EINVAL; |
| |
| rmidev->pid = input; |
| |
| if (rmidev->pid) { |
| rmidev->task = pid_task(find_vpid(rmidev->pid), PIDTYPE_PID); |
| if (!rmidev->task) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to locate PID of data logging tool\n", |
| __func__); |
| return -EINVAL; |
| } |
| } |
| |
| return count; |
| } |
| |
| static ssize_t rmidev_sysfs_term_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count) |
| { |
| unsigned int input; |
| |
| if (sscanf(buf, "%u", &input) != 1) |
| return -EINVAL; |
| |
| if (input != 1) |
| return -EINVAL; |
| |
| if (rmidev->pid) |
| send_sig_info(SIGTERM, &rmidev->terminate_signal, rmidev->task); |
| |
| return count; |
| } |
| |
| static ssize_t rmidev_sysfs_intr_mask_show(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| return snprintf(buf, PAGE_SIZE, "0x%02x\n", rmidev->intr_mask); |
| } |
| |
| static ssize_t rmidev_sysfs_intr_mask_store(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count) |
| { |
| unsigned int input; |
| |
| if (sscanf(buf, "%u", &input) != 1) |
| return -EINVAL; |
| |
| rmidev->intr_mask = (unsigned char)input; |
| |
| return count; |
| } |
| |
| /* |
| * rmidev_llseek - set register address to access for RMI device |
| * |
| * @filp: pointer to file structure |
| * @off: |
| * if whence == SEEK_SET, |
| * off: 16-bit RMI register address |
| * if whence == SEEK_CUR, |
| * off: offset from current position |
| * if whence == SEEK_END, |
| * off: offset from end position (0xFFFF) |
| * @whence: SEEK_SET, SEEK_CUR, or SEEK_END |
| */ |
| static loff_t rmidev_llseek(struct file *filp, loff_t off, int whence) |
| { |
| loff_t newpos; |
| struct rmidev_data *dev_data = filp->private_data; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (IS_ERR(dev_data)) { |
| pr_err("%s: Pointer of char device data is invalid", __func__); |
| return -EBADF; |
| } |
| |
| mutex_lock(&(dev_data->file_mutex)); |
| |
| switch (whence) { |
| case SEEK_SET: |
| newpos = off; |
| break; |
| case SEEK_CUR: |
| newpos = filp->f_pos + off; |
| break; |
| case SEEK_END: |
| newpos = REG_ADDR_LIMIT + off; |
| break; |
| default: |
| newpos = -EINVAL; |
| goto clean_up; |
| } |
| |
| if (newpos < 0 || newpos > REG_ADDR_LIMIT) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: New position 0x%04x is invalid\n", |
| __func__, (unsigned int)newpos); |
| newpos = -EINVAL; |
| goto clean_up; |
| } |
| |
| filp->f_pos = newpos; |
| |
| clean_up: |
| mutex_unlock(&(dev_data->file_mutex)); |
| |
| return newpos; |
| } |
| |
| /* |
| * rmidev_read: read register data from RMI device |
| * |
| * @filp: pointer to file structure |
| * @buf: pointer to user space buffer |
| * @count: number of bytes to read |
| * @f_pos: starting RMI register address |
| */ |
| static ssize_t rmidev_read(struct file *filp, char __user *buf, |
| size_t count, loff_t *f_pos) |
| { |
| ssize_t retval; |
| unsigned char tmpbuf[count + 1]; |
| struct rmidev_data *dev_data = filp->private_data; |
| |
| if (IS_ERR(dev_data)) { |
| pr_err("%s: Pointer of char device data is invalid", __func__); |
| return -EBADF; |
| } |
| |
| if (count == 0) |
| return 0; |
| |
| if (count > (REG_ADDR_LIMIT - *f_pos)) |
| count = REG_ADDR_LIMIT - *f_pos; |
| |
| mutex_lock(&(dev_data->file_mutex)); |
| |
| retval = synaptics_rmi4_reg_read(rmidev->rmi4_data, |
| *f_pos, |
| tmpbuf, |
| count); |
| if (retval < 0) |
| goto clean_up; |
| |
| if (copy_to_user(buf, tmpbuf, count)) |
| retval = -EFAULT; |
| else |
| *f_pos += retval; |
| |
| clean_up: |
| mutex_unlock(&(dev_data->file_mutex)); |
| |
| return retval; |
| } |
| |
| /* |
| * rmidev_write: write register data to RMI device |
| * |
| * @filp: pointer to file structure |
| * @buf: pointer to user space buffer |
| * @count: number of bytes to write |
| * @f_pos: starting RMI register address |
| */ |
| static ssize_t rmidev_write(struct file *filp, const char __user *buf, |
| size_t count, loff_t *f_pos) |
| { |
| ssize_t retval; |
| unsigned char tmpbuf[count + 1]; |
| struct rmidev_data *dev_data = filp->private_data; |
| |
| if (IS_ERR(dev_data)) { |
| pr_err("%s: Pointer of char device data is invalid", __func__); |
| return -EBADF; |
| } |
| |
| if (count == 0) |
| return 0; |
| |
| if (count > (REG_ADDR_LIMIT - *f_pos)) |
| count = REG_ADDR_LIMIT - *f_pos; |
| |
| if (copy_from_user(tmpbuf, buf, count)) |
| return -EFAULT; |
| |
| mutex_lock(&(dev_data->file_mutex)); |
| |
| retval = synaptics_rmi4_reg_write(rmidev->rmi4_data, |
| *f_pos, |
| tmpbuf, |
| count); |
| if (retval >= 0) |
| *f_pos += retval; |
| |
| mutex_unlock(&(dev_data->file_mutex)); |
| |
| return retval; |
| } |
| |
| static int rmidev_open(struct inode *inp, struct file *filp) |
| { |
| int retval = 0; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| struct rmidev_data *dev_data = |
| container_of(inp->i_cdev, struct rmidev_data, main_dev); |
| |
| if (!dev_data) |
| return -EACCES; |
| |
| filp->private_data = dev_data; |
| |
| mutex_lock(&(dev_data->file_mutex)); |
| |
| rmi4_data->irq_enable(rmi4_data, false, false); |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: Attention interrupt disabled\n", |
| __func__); |
| |
| if (dev_data->ref_count < 1) |
| dev_data->ref_count++; |
| else |
| retval = -EACCES; |
| |
| mutex_unlock(&(dev_data->file_mutex)); |
| |
| return retval; |
| } |
| |
| static int rmidev_release(struct inode *inp, struct file *filp) |
| { |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| struct rmidev_data *dev_data = |
| container_of(inp->i_cdev, struct rmidev_data, main_dev); |
| |
| if (!dev_data) |
| return -EACCES; |
| |
| mutex_lock(&(dev_data->file_mutex)); |
| |
| dev_data->ref_count--; |
| if (dev_data->ref_count < 0) |
| dev_data->ref_count = 0; |
| |
| rmi4_data->irq_enable(rmi4_data, true, false); |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: Attention interrupt enabled\n", |
| __func__); |
| |
| mutex_unlock(&(dev_data->file_mutex)); |
| |
| rmi4_data->reset_device(rmi4_data); |
| |
| return 0; |
| } |
| |
| static const struct file_operations rmidev_fops = { |
| .owner = THIS_MODULE, |
| .llseek = rmidev_llseek, |
| .read = rmidev_read, |
| .write = rmidev_write, |
| .open = rmidev_open, |
| .release = rmidev_release, |
| }; |
| |
| static void rmidev_device_cleanup(struct rmidev_data *dev_data) |
| { |
| dev_t devno; |
| struct synaptics_rmi4_data *rmi4_data = rmidev->rmi4_data; |
| |
| if (dev_data) { |
| devno = dev_data->main_dev.dev; |
| |
| if (dev_data->device_class) |
| device_destroy(dev_data->device_class, devno); |
| |
| cdev_del(&dev_data->main_dev); |
| |
| unregister_chrdev_region(devno, 1); |
| |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: rmidev device removed\n", |
| __func__); |
| } |
| |
| return; |
| } |
| |
| static char *rmi_char_devnode(struct device *dev, mode_t *mode) |
| { |
| if (!mode) |
| return NULL; |
| |
| *mode = (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP | S_IROTH | S_IWOTH); |
| |
| return kasprintf(GFP_KERNEL, "rmi/%s", dev_name(dev)); |
| } |
| |
| static int rmidev_create_device_class(void) |
| { |
| rmidev_device_class = class_create(THIS_MODULE, DEVICE_CLASS_NAME); |
| |
| if (IS_ERR(rmidev_device_class)) { |
| pr_err("%s: Failed to create /dev/%s\n", |
| __func__, CHAR_DEVICE_NAME); |
| return -ENODEV; |
| } |
| |
| rmidev_device_class->devnode = rmi_char_devnode; |
| |
| return 0; |
| } |
| |
| static void rmidev_attn(struct synaptics_rmi4_data *rmi4_data, |
| unsigned char intr_mask) |
| { |
| if (!rmidev) |
| return; |
| |
| if (rmidev->pid && (rmidev->intr_mask & intr_mask)) |
| send_sig_info(SIGIO, &rmidev->interrupt_signal, rmidev->task); |
| |
| return; |
| } |
| |
| static int rmidev_init_device(struct synaptics_rmi4_data *rmi4_data) |
| { |
| int retval; |
| dev_t dev_no; |
| unsigned char attr_count; |
| struct rmidev_data *dev_data; |
| struct device *device_ptr; |
| const struct synaptics_dsx_board_data *bdata = |
| rmi4_data->hw_if->board_data; |
| |
| rmidev = kzalloc(sizeof(*rmidev), GFP_KERNEL); |
| if (!rmidev) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to alloc mem for rmidev\n", |
| __func__); |
| retval = -ENOMEM; |
| goto err_rmidev; |
| } |
| |
| rmidev->rmi4_data = rmi4_data; |
| |
| memset(&rmidev->interrupt_signal, 0, sizeof(rmidev->interrupt_signal)); |
| rmidev->interrupt_signal.si_signo = SIGIO; |
| rmidev->interrupt_signal.si_code = SI_USER; |
| |
| memset(&rmidev->terminate_signal, 0, sizeof(rmidev->terminate_signal)); |
| rmidev->terminate_signal.si_signo = SIGTERM; |
| rmidev->terminate_signal.si_code = SI_USER; |
| |
| retval = rmidev_create_device_class(); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to create device class\n", |
| __func__); |
| goto err_device_class; |
| } |
| |
| if (rmidev_major_num) { |
| dev_no = MKDEV(rmidev_major_num, DEV_NUMBER); |
| retval = register_chrdev_region(dev_no, 1, CHAR_DEVICE_NAME); |
| } else { |
| retval = alloc_chrdev_region(&dev_no, 0, 1, CHAR_DEVICE_NAME); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to allocate char device region\n", |
| __func__); |
| goto err_device_region; |
| } |
| |
| rmidev_major_num = MAJOR(dev_no); |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: Major number of rmidev = %d\n", |
| __func__, rmidev_major_num); |
| } |
| |
| dev_data = kzalloc(sizeof(*dev_data), GFP_KERNEL); |
| if (!dev_data) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to alloc mem for dev_data\n", |
| __func__); |
| retval = -ENOMEM; |
| goto err_dev_data; |
| } |
| |
| mutex_init(&dev_data->file_mutex); |
| dev_data->rmi_dev = rmidev; |
| rmidev->data = dev_data; |
| |
| cdev_init(&dev_data->main_dev, &rmidev_fops); |
| |
| retval = cdev_add(&dev_data->main_dev, dev_no, 1); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to add rmi char device\n", |
| __func__); |
| goto err_char_device; |
| } |
| |
| dev_set_name(&rmidev->dev, "rmidev%d", MINOR(dev_no)); |
| dev_data->device_class = rmidev_device_class; |
| |
| device_ptr = device_create(dev_data->device_class, NULL, dev_no, |
| NULL, CHAR_DEVICE_NAME"%d", MINOR(dev_no)); |
| if (IS_ERR(device_ptr)) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to create rmi char device\n", |
| __func__); |
| retval = -ENODEV; |
| goto err_char_device; |
| } |
| |
| retval = gpio_export(bdata->irq_gpio, false); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to export attention gpio\n", |
| __func__); |
| } else { |
| retval = gpio_export_link(&(rmi4_data->input_dev->dev), |
| "attn", bdata->irq_gpio); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s Failed to create gpio symlink\n", |
| __func__); |
| } else { |
| dev_dbg(rmi4_data->pdev->dev.parent, |
| "%s: Exported attention gpio %d\n", |
| __func__, bdata->irq_gpio); |
| } |
| } |
| |
| rmidev->sysfs_dir = kobject_create_and_add(SYSFS_FOLDER_NAME, |
| &rmi4_data->input_dev->dev.kobj); |
| if (!rmidev->sysfs_dir) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to create sysfs directory\n", |
| __func__); |
| retval = -ENODEV; |
| goto err_sysfs_dir; |
| } |
| |
| retval = sysfs_create_bin_file(rmidev->sysfs_dir, |
| &attr_data); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to create sysfs bin file\n", |
| __func__); |
| goto err_sysfs_bin; |
| } |
| |
| for (attr_count = 0; attr_count < ARRAY_SIZE(attrs); attr_count++) { |
| retval = sysfs_create_file(rmidev->sysfs_dir, |
| &attrs[attr_count].attr); |
| if (retval < 0) { |
| dev_err(rmi4_data->pdev->dev.parent, |
| "%s: Failed to create sysfs attributes\n", |
| __func__); |
| retval = -ENODEV; |
| goto err_sysfs_attrs; |
| } |
| } |
| |
| return 0; |
| |
| err_sysfs_attrs: |
| for (attr_count--; attr_count >= 0; attr_count--) |
| sysfs_remove_file(rmidev->sysfs_dir, &attrs[attr_count].attr); |
| |
| sysfs_remove_bin_file(rmidev->sysfs_dir, &attr_data); |
| |
| err_sysfs_bin: |
| kobject_put(rmidev->sysfs_dir); |
| |
| err_sysfs_dir: |
| err_char_device: |
| rmidev_device_cleanup(dev_data); |
| kfree(dev_data); |
| |
| err_dev_data: |
| unregister_chrdev_region(dev_no, 1); |
| |
| err_device_region: |
| class_destroy(rmidev_device_class); |
| |
| err_device_class: |
| kfree(rmidev); |
| rmidev = NULL; |
| |
| err_rmidev: |
| return retval; |
| } |
| |
| static void rmidev_remove_device(struct synaptics_rmi4_data *rmi4_data) |
| { |
| unsigned char attr_count; |
| struct rmidev_data *dev_data; |
| |
| if (!rmidev) |
| goto exit; |
| |
| for (attr_count = 0; attr_count < ARRAY_SIZE(attrs); attr_count++) |
| sysfs_remove_file(rmidev->sysfs_dir, &attrs[attr_count].attr); |
| |
| sysfs_remove_bin_file(rmidev->sysfs_dir, &attr_data); |
| |
| kobject_put(rmidev->sysfs_dir); |
| |
| dev_data = rmidev->data; |
| if (dev_data) { |
| rmidev_device_cleanup(dev_data); |
| kfree(dev_data); |
| } |
| |
| unregister_chrdev_region(rmidev->dev_no, 1); |
| |
| class_destroy(rmidev_device_class); |
| |
| kfree(rmidev); |
| rmidev = NULL; |
| |
| exit: |
| complete(&rmidev_remove_complete); |
| |
| return; |
| } |
| |
| static struct synaptics_rmi4_exp_fn rmidev_module = { |
| .fn_type = RMI_DEV, |
| .init = rmidev_init_device, |
| .remove = rmidev_remove_device, |
| .reset = NULL, |
| .reinit = NULL, |
| .early_suspend = NULL, |
| .suspend = NULL, |
| .resume = NULL, |
| .late_resume = NULL, |
| .attn = rmidev_attn, |
| }; |
| |
| static int __init rmidev_module_init(void) |
| { |
| synaptics_rmi4_new_function(&rmidev_module, true); |
| |
| return 0; |
| } |
| |
| static void __exit rmidev_module_exit(void) |
| { |
| synaptics_rmi4_new_function(&rmidev_module, false); |
| |
| wait_for_completion(&rmidev_remove_complete); |
| |
| return; |
| } |
| |
| module_init(rmidev_module_init); |
| module_exit(rmidev_module_exit); |
| |
| MODULE_AUTHOR("Synaptics, Inc."); |
| MODULE_DESCRIPTION("Synaptics DSX RMI Dev Module"); |
| MODULE_LICENSE("GPL v2"); |