aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/misc/sec_jack_muic.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/misc/sec_jack_muic.c')
-rw-r--r--drivers/misc/sec_jack_muic.c461
1 files changed, 461 insertions, 0 deletions
diff --git a/drivers/misc/sec_jack_muic.c b/drivers/misc/sec_jack_muic.c
new file mode 100644
index 0000000..9450f1c
--- /dev/null
+++ b/drivers/misc/sec_jack_muic.c
@@ -0,0 +1,461 @@
+/* drivers/misc/sec_jack_muic.c
+ *
+ * Copyright (C) 2010 Samsung Electronics Co.Ltd
+ *
+ * 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/module.h>
+#include <linux/interrupt.h>
+#include <linux/workqueue.h>
+#include <linux/irq.h>
+#include <linux/delay.h>
+#include <linux/types.h>
+#include <linux/input.h>
+#include <linux/platform_device.h>
+#include <linux/errno.h>
+#include <linux/err.h>
+#include <linux/switch.h>
+#include <linux/input.h>
+#include <linux/timer.h>
+#include <linux/wakelock.h>
+#include <linux/slab.h>
+#include <linux/gpio.h>
+#include <linux/gpio_event.h>
+#include <linux/sec_jack.h>
+
+#include <plat/adc.h>
+
+/* keep this value if you support double-pressed concept */
+#if defined(CONFIG_TARGET_LOCALE_KOR)
+#define SEND_KEY_CHECK_TIME_MS 20 /* 20ms - GB VOC in KOR*/
+#elif defined(CONFIG_MACH_Q1_BD)
+/* 27ms, total delay is approximately double more
+ because hrtimer is called twice by gpio input driver,
+ new sec spec total delay is 60ms +/-10ms */
+#define SEND_KEY_CHECK_TIME_MS 27
+#else
+#define SEND_KEY_CHECK_TIME_MS 40 /* 40ms */
+#endif
+#define WAKE_LOCK_TIME (HZ * 5) /* 5 sec */
+#define WAKE_LOCK_TIME_IN_SENDKEY (HZ * 3)
+
+#define JACK_CLASS_NAME "audio"
+#define JACK_DEV_NAME "earjack"
+
+static struct class *jack_class;
+static struct device *jack_dev;
+
+struct sec_jack_info {
+ struct s3c_adc_client *padc;
+ struct sec_jack_platform_data *pdata;
+ struct delayed_work jack_detect_work;
+ struct work_struct buttons_work;
+ struct workqueue_struct *queue;
+ struct input_dev *input_dev;
+ struct wake_lock det_wake_lock;
+ struct sec_jack_zone *zone;
+ struct input_handler handler;
+ struct input_handle handle;
+ struct input_device_id ids[2];
+ int det_irq;
+ int dev_id;
+ int pressed;
+ int pressed_code;
+ struct platform_device *send_key_dev;
+ unsigned int cur_jack_type;
+ int det_status;
+};
+
+struct sec_jack_info *hi;
+static unsigned int send_end_pressed;
+
+/* with some modifications like moving all the gpio structs inside
+ * the platform data and getting the name for the switch and
+ * gpio_event from the platform data, the driver could support more than
+ * one headset jack, but currently user space is looking only for
+ * one key file and switch for a headset so it'd be overkill and
+ * untestable so we limit to one instantiation for now.
+ */
+static atomic_t instantiated = ATOMIC_INIT(0);
+
+/* sysfs name HeadsetObserver.java looks for to track headset state
+ */
+struct switch_dev switch_jack_detection = {
+ .name = "h2w",
+};
+
+/* To support AT+FCESTEST=1 */
+struct switch_dev switch_sendend = {
+ .name = "send_end",
+};
+
+static struct gpio_event_direct_entry sec_jack_key_map[] = {
+ {
+ .code = KEY_UNKNOWN,
+ },
+};
+
+static struct gpio_event_input_info sec_jack_key_info = {
+ .info.func = gpio_event_input_func,
+ .info.no_suspend = true,
+ .type = EV_KEY,
+ .debounce_time.tv64 = SEND_KEY_CHECK_TIME_MS * NSEC_PER_MSEC,
+ .keymap = sec_jack_key_map,
+ .keymap_size = ARRAY_SIZE(sec_jack_key_map)
+};
+
+static struct gpio_event_info *sec_jack_input_info[] = {
+ &sec_jack_key_info.info,
+};
+
+static struct gpio_event_platform_data sec_jack_input_data = {
+ .name = "sec_jack",
+ .info = sec_jack_input_info,
+ .info_count = ARRAY_SIZE(sec_jack_input_info),
+};
+
+static void sec_jack_set_type(struct sec_jack_info *hi, int jack_type)
+{
+ struct sec_jack_platform_data *pdata = hi->pdata;
+
+ /* this can happen during slow inserts where we think we identified
+ * the type but then we get another interrupt and do it again
+ */
+ if (jack_type == hi->cur_jack_type) {
+ if (jack_type != SEC_HEADSET_4POLE)
+ pdata->set_micbias_state(false);
+
+ return;
+ }
+
+ if (jack_type == SEC_HEADSET_4POLE) {
+ /* for a 4 pole headset, enable detection of send/end key */
+ if (hi->send_key_dev == NULL)
+ /* enable to get events again */
+ hi->send_key_dev = platform_device_register_data(NULL,
+ GPIO_EVENT_DEV_NAME,
+ hi->dev_id,
+ &sec_jack_input_data,
+ sizeof(sec_jack_input_data));
+ } else {
+ /* for all other jacks, disable send/end key detection */
+ if (hi->send_key_dev != NULL) {
+ /* disable to prevent false events on next insert */
+ platform_device_unregister(hi->send_key_dev);
+ hi->send_key_dev = NULL;
+ }
+ /* micbias is left enabled for 4pole and disabled otherwise */
+ pdata->set_micbias_state(false);
+ }
+ /* if user inserted ear jack slowly, different jack event can occur
+ * sometimes because irq_thread is defined IRQ_ONESHOT, detach status
+ * can be ignored sometimes so in that case, driver inform detach
+ * event to user side
+ */
+ switch_set_state(&switch_jack_detection, SEC_JACK_NO_DEVICE);
+
+ hi->cur_jack_type = jack_type;
+ pr_info("%s : jack_type = %d\n", __func__, jack_type);
+
+ switch_set_state(&switch_jack_detection, jack_type);
+}
+
+static void handle_jack_not_inserted(struct sec_jack_info *hi)
+{
+ sec_jack_set_type(hi, SEC_JACK_NO_DEVICE);
+ hi->pdata->set_micbias_state(false);
+}
+
+static ssize_t select_jack_show(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ pr_info("%s : operate nothing\n", __func__);
+
+ return 0;
+}
+
+static ssize_t select_jack_store(struct device *dev,
+ struct device_attribute *attr, const char *buf, size_t size)
+{
+ struct sec_jack_info *hi = dev_get_drvdata(dev);
+ struct sec_jack_platform_data *pdata = hi->pdata;
+ int value = 0;
+
+
+ sscanf(buf, "%d", &value);
+ pr_err("%s: User selection : 0X%x", __func__, value);
+ if (value == SEC_HEADSET_4POLE) {
+ pdata->set_micbias_state(true);
+ msleep(100);
+ }
+
+ sec_jack_set_type(hi, value);
+
+ return size;
+}
+
+static ssize_t earjack_key_state_show(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ struct sec_jack_info *hi = dev_get_drvdata(dev);
+ int value = 0;
+
+ if (hi->pressed <= 0)
+ value = 0;
+ else
+ value = 1;
+
+ return sprintf(buf, "%d\n", value);
+}
+
+static ssize_t earjack_key_state_store(struct device *dev,
+ struct device_attribute *attr, const char *buf, size_t size)
+{
+ pr_info("%s : operate nothing\n", __func__);
+
+ return size;
+}
+
+static ssize_t earjack_state_show(struct device *dev,
+ struct device_attribute *attr, char *buf)
+{
+ struct sec_jack_info *hi = dev_get_drvdata(dev);
+ int value = 0;
+
+ if (hi->cur_jack_type == SEC_HEADSET_4POLE)
+ value = 1;
+ else
+ value = 0;
+
+ return sprintf(buf, "%d\n", value);
+}
+
+static ssize_t earjack_state_store(struct device *dev,
+ struct device_attribute *attr, const char *buf, size_t size)
+{
+ pr_info("%s : operate nothing\n", __func__);
+
+ return size;
+}
+static DEVICE_ATTR(key_state, S_IRUGO | S_IWUSR | S_IWGRP,
+ earjack_key_state_show, earjack_key_state_store);
+
+static DEVICE_ATTR(state, S_IRUGO | S_IWUSR | S_IWGRP,
+ earjack_state_show, earjack_state_store);
+
+static DEVICE_ATTR(select_jack, S_IRUGO | S_IWUSR | S_IWGRP,
+ select_jack_show, select_jack_store);
+
+void jack_status_change(int status)
+{
+ int jack_type = SEC_JACK_NO_DEVICE;
+
+ pr_info("%s status=%d\n", __func__, status);
+
+ wake_lock_timeout(&hi->det_wake_lock, WAKE_LOCK_TIME);
+
+ if (status) {
+ jack_type = SEC_HEADSET_4POLE;
+ } else {
+ jack_type = SEC_JACK_NO_DEVICE;
+ if (send_end_pressed == 1) {
+ input_report_key(hi->input_dev, KEY_MEDIA, 0);
+ input_sync(hi->input_dev);
+ switch_set_state(&switch_sendend, 0);
+ send_end_pressed = 0;
+ }
+ }
+ switch_set_state(&switch_jack_detection, jack_type);
+ hi->cur_jack_type = jack_type;
+}
+EXPORT_SYMBOL(jack_status_change);
+
+void earkey_status_change(int pressed, int kcode)
+{
+ wake_lock_timeout(&hi->det_wake_lock, WAKE_LOCK_TIME_IN_SENDKEY);
+ hi->pressed_code = kcode;
+ if (pressed) {
+ input_report_key(hi->input_dev, hi->pressed_code, 1);
+ switch_set_state(&switch_sendend, 1);
+ input_sync(hi->input_dev);
+ send_end_pressed = 1;
+ } else {
+ input_report_key(hi->input_dev, hi->pressed_code, 0);
+ switch_set_state(&switch_sendend, 0);
+ input_sync(hi->input_dev);
+ send_end_pressed = 0;
+ }
+ hi->pressed = send_end_pressed;
+}
+EXPORT_SYMBOL(earkey_status_change);
+
+static int sec_jack_probe(struct platform_device *pdev)
+{
+ struct sec_jack_platform_data *pdata = pdev->dev.platform_data;
+ int ret;
+
+ pr_info("%s : Registering jack driver\n", __func__);
+ if (!pdata) {
+ pr_err("%s : pdata is NULL.\n", __func__);
+ return -ENODEV;
+ }
+
+ if (atomic_xchg(&instantiated, 1)) {
+ pr_err("%s : already instantiated, can only have one\n",
+ __func__);
+ return -ENODEV;
+ }
+
+ hi = kzalloc(sizeof(struct sec_jack_info), GFP_KERNEL);
+ if (hi == NULL) {
+ pr_err("%s : Failed to allocate memory.\n", __func__);
+ ret = -ENOMEM;
+ goto err_kzalloc;
+ }
+
+ hi->pdata = pdata;
+ send_end_pressed = 0;
+
+ /* make the id of our gpio_event device the same as our platform device,
+ * which makes it the responsiblity of the board file to make sure
+ * it is unique relative to other gpio_event devices
+ */
+ hi->dev_id = pdev->id;
+
+ hi->input_dev = input_allocate_device();
+ if (hi->input_dev == NULL) {
+ ret = -ENOMEM;
+ pr_err("%s : Failed to allocate input device.\n", __func__);
+ }
+ pr_info("%s input device is [%s]\n", __func__, hi->input_dev->name);
+
+ hi->input_dev->name = "sec_jack";
+ input_set_capability(hi->input_dev , EV_KEY, KEY_MEDIA);
+ input_set_capability(hi->input_dev , EV_KEY, KEY_VOLUMEDOWN);
+ input_set_capability(hi->input_dev , EV_KEY, KEY_VOLUMEUP);
+ ret = input_register_device(hi->input_dev);
+ if (ret) {
+ pr_err("%s : Failed to register driver\n", __func__);
+ goto err_register_input_dev;
+ }
+
+ ret = switch_dev_register(&switch_jack_detection);
+ if (ret < 0) {
+ pr_err("%s : Failed to register switch device\n", __func__);
+ goto err_register_input_dev;
+ }
+
+ ret = switch_dev_register(&switch_sendend);
+ if (ret < 0) {
+ printk(KERN_ERR "SEC JACK: Failed to register switch device\n");
+ goto err_switch_dev_register_send_end;
+ }
+ wake_lock_init(&hi->det_wake_lock, WAKE_LOCK_SUSPEND, "sec_jack_det");
+
+ hi->queue = create_singlethread_workqueue("sec_jack_wq");
+ if (hi->queue == NULL) {
+ ret = -ENOMEM;
+ pr_err("%s: Failed to create workqueue\n", __func__);
+ goto err_create_wq_failed;
+ }
+
+ jack_class = class_create(THIS_MODULE, JACK_CLASS_NAME);
+ if (IS_ERR(jack_class))
+ pr_err("Failed to create class(sec_jack)\n");
+
+ /* support PBA function test */
+ jack_dev = device_create(jack_class, NULL, 0, hi, JACK_DEV_NAME);
+ if (IS_ERR(jack_dev))
+ pr_err("Failed to create device(sec_jack)!= %ld\n",
+ IS_ERR(jack_dev));
+
+ if (device_create_file(jack_dev, &dev_attr_select_jack) < 0)
+ pr_err("Failed to create device file(%s)!\n",
+ dev_attr_select_jack.attr.name);
+
+ if (device_create_file(jack_dev, &dev_attr_key_state) < 0)
+ pr_err("Failed to create device file (%s)!\n",
+ dev_attr_key_state.attr.name);
+
+ if (device_create_file(jack_dev, &dev_attr_state) < 0)
+ pr_err("Failed to create device file (%s)!\n",
+ dev_attr_state.attr.name);
+
+ return 0;
+
+err_create_wq_failed:
+ destroy_workqueue(hi->queue);
+ wake_lock_destroy(&hi->det_wake_lock);
+ switch_dev_unregister(&switch_sendend);
+err_switch_dev_register_send_end:
+ switch_dev_unregister(&switch_jack_detection);
+err_register_input_dev:
+ input_free_device(hi->input_dev);
+err_kzalloc:
+ atomic_set(&instantiated, 0);
+
+ return ret;
+}
+
+static int sec_jack_remove(struct platform_device *pdev)
+{
+
+ struct sec_jack_info *hi = dev_get_drvdata(&pdev->dev);
+
+ pr_info("%s :\n", __func__);
+ destroy_workqueue(hi->queue);
+ if (hi->send_key_dev) {
+ platform_device_unregister(hi->send_key_dev);
+ hi->send_key_dev = NULL;
+ }
+ input_unregister_handler(&hi->handler);
+ wake_lock_destroy(&hi->det_wake_lock);
+ switch_dev_unregister(&switch_sendend);
+ switch_dev_unregister(&switch_jack_detection);
+ kfree(hi);
+ atomic_set(&instantiated, 0);
+
+ return 0;
+}
+
+
+static struct platform_driver sec_jack_driver = {
+ .probe = sec_jack_probe,
+ .remove = sec_jack_remove,
+ .driver = {
+ .name = "sec_jack",
+ .owner = THIS_MODULE,
+ },
+};
+
+static int __init sec_jack_init(void)
+{
+ int ret;
+
+ ret = platform_driver_register(&sec_jack_driver);
+
+ if (ret)
+ pr_err("%s: Failed to add sec jack driver\n", __func__);
+
+ return ret;
+}
+
+static void __exit sec_jack_exit(void)
+{
+ platform_driver_unregister(&sec_jack_driver);
+}
+
+module_init(sec_jack_init);
+module_exit(sec_jack_exit);
+
+MODULE_AUTHOR("sopia.kim@samsung.com");
+MODULE_DESCRIPTION("Samsung Electronics Corp Ear-Jack detection driver");
+MODULE_LICENSE("GPL");