android_kernel_cmhtcleo/arch/arm/mach-msm/htc_headset_gpio.c

463 lines
10 KiB
C
Raw Normal View History

2010-08-27 09:19:57 +00:00
/*
*
* /arch/arm/mach-msm/htc_headset_gpio.c
*
* HTC GPIO headset detection driver.
*
* Copyright (C) 2010 HTC, Inc.
*
* This software is licensed under the terms of the GNU General Public
* License version 2, as published by the Free Software Foundation, and
* may be copied, distributed, and modified under those terms.
*
* 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/sysdev.h>
#include <linux/device.h>
#include <linux/interrupt.h>
#include <linux/irq.h>
#include <linux/types.h>
#include <linux/input.h>
#include <linux/platform_device.h>
#include <linux/errno.h>
#include <linux/err.h>
#include <linux/hrtimer.h>
#include <linux/workqueue.h>
#include <linux/wakelock.h>
#include <linux/delay.h>
#include <linux/gpio.h>
#include <mach/atmega_microp.h>
2010-08-27 09:19:57 +00:00
#include <mach/htc_headset_mgr.h>
#include <mach/htc_headset_gpio.h>
#define DRIVER_NAME "HS_GPIO"
/* #define DEBUG */
#ifdef DEBUG
#define AJ_DBG(fmt, arg...) \
printk(KERN_INFO "[Audio Jack] %s " fmt "\r\n", __func__, ## arg)
#else
#define AJ_DBG(fmt, arg...) do {} while (0)
#endif
int microp_get_remote_adc(uint32_t *val);
int microp_set_adc_req(uint8_t channel);
2010-08-27 09:19:57 +00:00
struct audio_jack_info {
unsigned int irq_jack;
unsigned int irq_mic;
2010-08-27 09:19:57 +00:00
int audio_jack_detect;
int key_enable_gpio;
int mic_select_gpio;
int audio_jack_flag;
int mic_detect;
int last_pressed_key;
int microp_channel;
2010-08-27 09:19:57 +00:00
struct hrtimer detection_timer;
ktime_t debounce_time;
struct work_struct work;
struct work_struct mic_work;
2010-08-27 09:19:57 +00:00
spinlock_t spin_lock;
struct wake_lock audiojack_wake_lock;
};
static struct audio_jack_info *pjack_info;
int microp_set_adc_req(uint8_t value)
{
int ret;
uint8_t cmd[1];
cmd[0] = value; //value; TODO finish code... now only keys ADC
ret = microp_i2c_write(MICROP_I2C_WCMD_ADC_REQ, cmd, 1);
if (ret < 0)
{
pr_err("%s: request adc fail\n", __func__);
return -EIO;
}
return 0;
}
int microp_get_remote_adc(uint32_t *val)
{
int ret;
uint8_t data[4];
if (!val)
return -EIO;
ret = microp_i2c_read(MICROP_I2C_RCMD_ADC_VALUE, data, 2);
if (ret < 0)
{
pr_err("%s: request adc fail\n", __func__);
return -EIO;
}
// printk("%x %x\n", data[0], data[1]);
*val = data[1] | (data[0] << 8);
printk("remote adc %d\n", *val);
return 0;
}
static int hs_gpio_get_mic(void)
{
int value;
value = !gpio_get_value(pjack_info->mic_detect);
printk("hs_gpio_get_mic: %d\n", value);
return value;
}
static int hs_enable_key_irq(int status)
{
printk("hs_enable_key_irq: %d\n", status);
if(status)
enable_irq(pjack_info->irq_mic);
else
disable_irq(pjack_info->irq_mic);
return 0;
}
2010-08-27 09:19:57 +00:00
void hs_gpio_key_enable(int enable)
{
DBG_MSG();
if (pjack_info->key_enable_gpio)
gpio_set_value(pjack_info->key_enable_gpio, enable);
}
void hs_gpio_mic_select(int enable)
{
DBG_MSG();
if (pjack_info->mic_select_gpio)
gpio_set_value(pjack_info->mic_select_gpio, enable);
}
static int get_remote_keycode(int *keycode)
{
uint32_t val;
uint32_t btn = 0;
microp_set_adc_req(pjack_info->microp_channel);
if (microp_get_remote_adc(&val))
{
// failed. who know why? ignore
*keycode = 0;
return 1;
}
if((val >= 0) && (val <= 33))
{
btn = 1;
}
else if((val >= 38) && (val <= 82))
{
btn = 2;
}
else if((val >= 95) && (val <= 200))
{
btn = 3;
}
else if(val > 200)
{
// check previous key
if (pjack_info->last_pressed_key)
{
*keycode = pjack_info->last_pressed_key | 0x80;
pjack_info->last_pressed_key = 0;
return 0;
}
*keycode = 0;
return 1;
}
pjack_info->last_pressed_key = btn;
*keycode = btn;
return 0;
}
static irqreturn_t mic_irq_handler(int irq, void *dev_id)
{
pr_info("MIC IRQ Handler\n");
int value1, value2;
int retry_limit = 10;
AJ_DBG("");
do {
value1 = gpio_get_value(pjack_info->mic_detect);
set_irq_type(pjack_info->irq_mic, value1 ? IRQF_TRIGGER_LOW : IRQF_TRIGGER_HIGH);
value2 = gpio_get_value(pjack_info->mic_detect);
} while (value1 != value2 && retry_limit-- > 0);
AJ_DBG("value2 = %d (%d retries)", value2, (10-retry_limit));
schedule_work(&pjack_info->mic_work);
return IRQ_HANDLED;
}
2010-08-27 09:19:57 +00:00
static irqreturn_t detect_irq_handler(int irq, void *dev_id)
{
pr_info("DET IRQ Handler\n");
2010-08-27 09:19:57 +00:00
int value1, value2;
int retry_limit = 10;
hs_notify_hpin_irq();
AJ_DBG("");
do {
value1 = gpio_get_value(pjack_info->audio_jack_detect);
set_irq_type(pjack_info->irq_jack, value1 ?
IRQF_TRIGGER_LOW : IRQF_TRIGGER_HIGH);
value2 = gpio_get_value(pjack_info->audio_jack_detect);
} while (value1 != value2 && retry_limit-- > 0);
AJ_DBG("value2 = %d (%d retries)", value2, (10-retry_limit));
if ((pjack_info->audio_jack_flag == 0) ^ value2) {
wake_lock_timeout(&pjack_info->audiojack_wake_lock, 4*HZ);
/* Do the rest of the work in timer context */
hrtimer_start(&pjack_info->detection_timer,
pjack_info->debounce_time, HRTIMER_MODE_REL);
}
return IRQ_HANDLED;
}
static enum hrtimer_restart detect_35mm_event_timer_func(struct hrtimer *data)
{
int state;
AJ_DBG("");
state = !gpio_get_value(pjack_info->audio_jack_detect);
if (pjack_info->audio_jack_flag != state) {
pjack_info->audio_jack_flag = state;
schedule_work(&pjack_info->work);
}
return HRTIMER_NORESTART;
}
static void mic_work_func(struct work_struct *work)
{
pr_info("MIC Schedule Work\n");
int keycode = 0;
printk("mic_intr_work_func\n");
if (get_remote_keycode(&keycode) == 0)
{
printk("keycode %d\n", keycode);
htc_35mm_remote_notify_button_status(keycode);
}
else
printk("mic error keycode\n");
}
2010-08-27 09:19:57 +00:00
static void audiojack_work_func(struct work_struct *work)
{
int is_insert;
pr_info("DET Schedule Work\n");
2010-08-27 09:19:57 +00:00
unsigned long flags = 0;
spin_lock_irqsave(&pjack_info->spin_lock, flags);
is_insert = pjack_info->audio_jack_flag;
spin_unlock_irqrestore(&pjack_info->spin_lock, flags);
htc_35mm_remote_notify_insert_ext_headset(is_insert);
if (is_insert)
pjack_info->debounce_time = ktime_set(0, 200000000);
else
pjack_info->debounce_time = ktime_set(0, 500000000);
}
static void hs_gpio_register(void)
{
struct headset_notifier notifier;
if (pjack_info->mic_select_gpio) {
notifier.id = HEADSET_REG_MIC_SELECT;
notifier.func = hs_gpio_mic_select;
headset_notifier_register(&notifier);
}
if (pjack_info->key_enable_gpio) {
notifier.id = HEADSET_REG_KEY_ENABLE;
notifier.func = hs_gpio_key_enable;
headset_notifier_register(&notifier);
}
if (pjack_info->mic_detect) {
notifier.id = HEADSET_REG_MIC_STATUS;
notifier.func = hs_gpio_get_mic;
headset_notifier_register(&notifier);
notifier.id = HEADSET_REG_KEY_INT_ENABLE;
notifier.func = hs_enable_key_irq;
headset_notifier_register(&notifier);
}
2010-08-27 09:19:57 +00:00
}
static int audiojack_probe(struct platform_device *pdev)
{
int ret;
struct htc_headset_gpio_platform_data *pdata = pdev->dev.platform_data;
SYS_MSG("++++++++++++++++++++");
pjack_info = kzalloc(sizeof(struct audio_jack_info), GFP_KERNEL);
if (!pjack_info)
return -ENOMEM;
pjack_info->audio_jack_detect = pdata->hpin_gpio;
pjack_info->key_enable_gpio = pdata->key_enable_gpio;
pjack_info->mic_select_gpio = pdata->mic_select_gpio;
pjack_info->mic_detect = pdata->mic_detect_gpio;
pjack_info->microp_channel = pdata->microp_channel;
2010-08-27 09:19:57 +00:00
pjack_info->audio_jack_flag = 0;
pjack_info->last_pressed_key = 0;
2010-08-27 09:19:57 +00:00
pjack_info->debounce_time = ktime_set(0, 500000000);
hrtimer_init(&pjack_info->detection_timer,
CLOCK_MONOTONIC, HRTIMER_MODE_REL);
pjack_info->detection_timer.function = detect_35mm_event_timer_func;
INIT_WORK(&pjack_info->work, audiojack_work_func);
INIT_WORK(&pjack_info->mic_work, mic_work_func);
2010-08-27 09:19:57 +00:00
spin_lock_init(&pjack_info->spin_lock);
wake_lock_init(&pjack_info->audiojack_wake_lock,
WAKE_LOCK_SUSPEND, "audiojack");
if (pjack_info->audio_jack_detect) {
ret = gpio_request(pjack_info->audio_jack_detect,
"3.5mm_detect");
if (ret < 0)
goto err_request_detect_gpio;
ret = gpio_direction_input(pjack_info->audio_jack_detect);
if (ret < 0)
goto err_set_detect_gpio;
pjack_info->irq_jack =
gpio_to_irq(pjack_info->audio_jack_detect);
if (pjack_info->irq_jack < 0) {
ret = pjack_info->irq_jack;
goto err_request_detect_irq;
}
ret = request_irq(pjack_info->irq_jack,
detect_irq_handler,
IRQF_TRIGGER_LOW, "35mm_headset", NULL);
if (ret < 0)
goto err_request_detect_irq;
ret = set_irq_wake(pjack_info->irq_jack, 1);
if (ret < 0)
goto err_set_irq_wake;
pr_info("DET IRQ Registered!");
2010-08-27 09:19:57 +00:00
}
if (pjack_info->mic_detect) {
ret = gpio_request(pjack_info->mic_detect,
"mic_detect");
if (ret < 0)
goto err_request_detect_gpio;
ret = gpio_direction_input(pjack_info->mic_detect);
if (ret < 0)
goto err_set_detect_gpio;
pjack_info->irq_mic =
gpio_to_irq(pjack_info->mic_detect);
if (pjack_info->irq_mic < 0) {
ret = pjack_info->irq_mic;
goto err_request_detect_irq;
}
ret = request_irq(pjack_info->irq_mic,
mic_irq_handler, IRQF_DISABLED | IRQF_TRIGGER_LOW, "mic_headset", NULL);
if (ret < 0)
goto err_request_detect_irq;
disable_irq(pjack_info->irq_mic);
pr_info("MIC IRQ Registered!");
}
2010-08-27 09:19:57 +00:00
hs_gpio_register();
SYS_MSG("--------------------");
return 0;
err_set_irq_wake:
if (pjack_info->audio_jack_detect)
free_irq(pjack_info->irq_jack, 0);
err_request_detect_irq:
err_set_detect_gpio:
if (pjack_info->audio_jack_detect)
gpio_free(pjack_info->audio_jack_detect);
err_request_detect_gpio:
printk(KERN_ERR "Audiojack: Failed in audiojack_probe\n");
return ret;
}
static int audiojack_remove(struct platform_device *pdev)
{
if (pjack_info->audio_jack_detect)
free_irq(pjack_info->irq_jack, 0);
if (pjack_info->audio_jack_detect)
free_irq(pjack_info->irq_mic, 0);
2010-08-27 09:19:57 +00:00
if (pjack_info->audio_jack_detect)
gpio_free(pjack_info->audio_jack_detect);
if (pjack_info->audio_jack_detect)
gpio_free(pjack_info->mic_detect);
2010-08-27 09:19:57 +00:00
return 0;
}
static struct platform_driver audiojack_driver = {
.probe = audiojack_probe,
.remove = audiojack_remove,
.driver = {
.name = "HTC_HEADSET_GPIO",
.owner = THIS_MODULE,
},
};
static int __init audiojack_init(void)
{
return platform_driver_register(&audiojack_driver);
}
static void __exit audiojack_exit(void)
{
platform_driver_unregister(&audiojack_driver);
}
module_init(audiojack_init);
module_exit(audiojack_exit);
MODULE_DESCRIPTION("HTC GPIO headset detection driver");
MODULE_LICENSE("GPL");