转自CSDN:http://blog.csdn.net/wlwl0071986/article/details/9154541
一、HALL开关原理及手机应用
手机中用来控制线路通断的器件主要有三种类型:开关、干簧管和霍尔元件。不同的是开关一般是由人工
手动控制,而干簧管和霍尔元件则通过磁信号来控制线路的通与断。霍尔(HALL)传感器是一种电子元件,其
外型封装和三极管非常相象。它是由HALL元件,放大器、施密特电路以及集电极开路输出三极管组成,当磁场
作用于HALL元件时产生一个微小的HALL电压,经放大器放大和施密特电路后使三极管导通输出低电平,而没
有磁场作用的时候(即翻盖打开后)三极管截止输出为高电平。和干簧管相比HALL传感器寿命更长,不容易损
坏,而且对振动,加速度不太敏感,作用时开关时间也比较快,通常为0.1~~2ms比干簧管的1~~3ms快得多。
手机中HALL传感器由一个开关型HALL元件和两个电源开关控制管组成。其导通与否完全受到手机CPU输
出的HALL高电平信号控制,电源则来自于电池。当翻盖合上时装在翻盖中的磁铁的磁场作用于HALL传感器
(一般翻盖/折叠手机都把磁铁安装在翻盖上),HALL电路中的三极管导通,从传感器的引脚输出低电平,如果
是在通话后则作为“挂机”信号送给CPU挂机。(这也就是为什么合上翻盖后手机就挂断的道理)。
当用户打开翻盖时,HALL不受磁场感应,HALL电路中的三极管截止,输出为高电平,如果该信号是在来
电时产生的,那么在送给CPU时,CPU便作为开机信号而接听电话。但如果仅仅是用户做其他操作比如输入短
信,电话号码单纯打开翻盖,该电路信号由CPU作为背景灯控制信号使背景灯点亮。(每次开盖的时候背景灯
都要点亮,同时记录一次翻盖次数)。小心的用工具仅仅掀开一点点翻盖的时候背景灯是不会亮的,因为这时
候还有磁场作用于HALL元件,当打开到一定角度的时候,失去磁场作用的HALL电路的三极管便截止输入高电
平,CPU在收到该信号后便驱动背景灯电路点燃背景灯。
当用户取消“翻盖接听”的选项后,CPU送出的HALL信号为低电平,从而使那两个电源开关控制管截止,没
有电源供给,即使在有无磁场信号时输出的电压都不会改变,因而也就失去了开关的作用。因此在这样的情况
下,来电后你翻盖CPU根据设置并不接通电话,这时候你需要按下接听键才能接听。
二、硬件连接
NOTE: 当有磁场改变的时候在OUT脚位会输出一个 H 或者 L 电平,与OUT 相连的PIN脚必须是一个可唤醒系统的中断
三、HALL KEY实现
在驱动gpio_event.c里添加中断注册和回掉函数
#define GN_HALL_KEY_OPEN 111 #define GN_HALL_KEY_CLOSE 112 -------给系统上报的键值 #define GN_GPIO_HALL_EINT_PIN 40 u8 hall_key_state_bak; spinlock_t hall_lock; struct hall_irq_info { int irq; struct work_struct work; }; struct hall_irq_info hall_info; enum hall_state { HALL_CLOSE = 0, HALL_OPEN =1 }; INIT_WORK(&(hall_info.work), work_func); ---------------初始化工作队列 err = gpio_request(GN_GPIO_HALL_EINT_PIN, "Hall Key IRQ GPIO"); if (err) { printk(KERN_ERR "%s: Failed to request GPIO %d\n", __func__, GN_GPIO_HALL_EINT_PIN); goto err_request_gpio_failed; } gpio_tlmm_config(GPIO_CFG(GN_GPIO_HALL_EINT_PIN, 0, GPIO_CFG_INPUT, GPIO_CFG_PULL_UP, GPIO_CFG_6MA),GPIO_CFG_ENABLE); gpio_free(GN_GPIO_HALL_EINT_PIN); hall_info.irq= gpio_to_irq(GN_GPIO_HALL_EINT_PIN); if (__gpio_get_value(GN_GPIO_HALL_EINT_PIN) == 0) { pr_info("HALL_KEY: init the request irq for RISING\n"); hall_key_state_bak = 0; switch_set_state((struct switch_dev *)&hall_data, HALL_CLOSE); err = request_irq(hall_info.irq, hall_irq, IRQF_TRIGGER_HIGH, "Hall_Key", NULL); } else { pr_info("HALL_KEY: init the request irq for FALLING\n"); hall_key_state_bak = 1; switch_set_state((struct switch_dev *)&hall_data, HALL_OPEN); err = request_irq(hall_info.irq, hall_irq, IRQF_TRIGGER_LOW, "Hall_Key", NULL); } enable_irq_wake(hall_info.irq); if (err < 0) { pr_err("HALL_KEY: enable_irq_wake failed...\n"); goto err_request_irq_failed; } -------------中断的配置 static void work_func(struct work_struct *work) { - - if (kpd_hallkey_state==1) { //hall open input_report_key(hall_input_dev, GN_HALL_KEY_OPEN, 1); input_sync(hall_input_dev); input_report_key(hall_input_dev, GN_HALL_KEY_OPEN, 0); input_sync(hall_input_dev); switch_set_state((struct switch_dev *)&hall_data, HALL_OPEN); } else {//hall close input_report_key(hall_input_dev, GN_HALL_KEY_CLOSE, 1); input_sync(hall_input_dev); input_report_key(hall_input_dev, GN_HALL_KEY_CLOSE, 0); input_sync(hall_input_dev); switch_set_state((struct switch_dev *)&hall_data, HALL_CLOSE); } - - } static irqreturn_t hall_irq(int irq, void *dev_id) { disable_irq_nosync(hall_info.irq); schedule_work(&hall_info.work); return IRQ_HANDLED; }
四、按键唤醒系统
要实现按键唤醒系统,首先,在驱动中,设置此键会被上报。如:
__set_bit(KEY_2, kpd_input_dev->keybit); __set_bit(KEY_3, kpd_input_dev->keybit);
这样,才能确保按键被注册处理,否则,上层收不到按键事件。
另外,在7x27a_kp.kl文件中,设置devices\qcom\msm7627a_sku3\7x27a_kpd.kl
key 111 GN_HALL_KEY_OPEN WAKE key 112 GN_HALL_KEY_CLOSE
这样就可实现按键唤醒系统了。
五、新增两个按键实现hall key
1. 首先修改底层gpio_event.c文件中的按键定义:
#define GN_HALL_KEY_OPEN 111 #define GN_HALL_KEY_CLOSE 112
2. 修改devices\qcom\msm7627a_sku3\7x27a_kpd.kl文件,增加
key 111 GN_HALL_KEY_OPEN WAKE (打开可以唤醒) key 112 GN_HALL_KEY_CLOSE
3. 在frameworks\base\include\ui\keycodelabels.h KEYCODES结构体中增加
{ "GN_HALL_KEY_OPEN", 111 }, { "GN_HALL_KEY_CLOSE", 112 },
4. 在frameworks\base\core\java\android\view\keyevent.java构造函数中增加:
public static final int KEYCODE_GN_HALL_KEY_OPEN = 111; public static final int KEYCODE_ GN_HALL_KEY_CLOSE = 112;
5. 修改: private static final int LAST_KEYCODE = KEYCODE_BUTTON_MODE;
为private static final int LAST_KEYCODE =KEYCODE_ GN_HALL_KEY_CLOSE;
在frameworks\base\native\include\android\keycodes.h的enum中增加:
AKEYCODE_GN_HALL_KEY_CLOSE = 250, AKEYCODE_GN_HALL_KEY_CLOSE = 251,
6. 在frameworks\base\libs\ui\input.cpp的issystemkey中增加:
case AKEYCODE_GN_HALL_KEY_CLOSE:
case AKEYCODE_GN_HALL_KEY_CLOSE:
这样bool KeyEvent::isSystemKey() const {
return isSystemKey(getKeyCode());
才能由scancode转换为keycode供上层应用截取。
7. 在frameworks/base/core/res/res/values/attrs.xml中增加:
<enum name="KEYCODE_GN_HALL_KEY_OPEN" value="250" /> <enum name="KEYCODE_GN_HALL_KEY_CLOSE" value="251" />