android手机霍尔开关的介绍

转自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根据设置并不接通电话,这时候你需要按下接听键才能接听。

二、硬件连接

 

android手机霍尔开关的介绍_第1张图片

 

 

 

 

 

 

            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中增加:


 

你可能感兴趣的:(Android)