基于QCM 8960
二十七、Audio jack
1. https://wiki.sonyericsson.net/androiki/Huashan_legacy_feature:_Vibrator_%26_Audio_Jack2. How to get debug info
1 dump register before and after headset plug, use following command to dump register
1.1 $ adb shell 'mount -t debugfs debugfs /sys/kernel/debug/'
1.2 $ adb shell 'cat /sys/kernel/debug/asoc/msm8960-snd-card/tabla_codec/codec_reg'
2 Log message during insertion please enable wcd9xxx-core.c +p, wcd9310.c +p, wcd9xxx-irq.c like following
2.1 $ adb shell 'mount -t debugfs debugfs /sys/kernel/debug/'
2.2 $ adb shell 'echo -n "file wcd9310.c +p" > /sys/kernel/debug/dynamic_debug/control'
2.3 $ adb shell 'echo -n "file wcd9xxx-core.c +p" > /sys/kernel/debug/dynamic_debug/control'
2.4 $ adb shell 'echo -n "file wcd9xxx-irq.c +p" > /sys/kernel/debug/dynamic_debug/control'
4. $ adb shell 'cat /sys/kernel/debug/gpio' 得到GPIO状态 gpio-189 = 151 + PMIC8921的38 Pin
gpio-189 (-- ) in lo 0x05 0x10 0x22 0x30 0x40 0x58
5. cat sys/module/snd_soc_msm8960/parameters/hs_detect_use_gpio; 可查看是否使用gpio中断检测headset insert/remove
二十八、如何检测 headset/headphone 插入或拔出
0. 针对wcd9310不需要读取firmware.
1. 主要分析 /kernel/sound/soc/codecs/wcd9310.c, kernel/sound/soc/msm/msm8960.c
2. msm8960_audrx_init@msm8960.c中 全局变量hs_detect_use_gpio 设置是否用PMIC的GPIO检测耳机插入
err = tabla_hs_detect(codec, &mbhc_cfg); //设置GPIO38的中断处理函数
tabla_hs_detect->tabla_mbhc_init_and_calibrate 设置中断处理函数 tabla_mechanical_plug_detect_irq,
中断pin是tabla->mbhc_cfg.gpio_irq;
@kernel/sound/soc/msm/msm8960.c中找到中断Pin的定义msm8960_audrx_init函数中 mbhc_cfg.gpio_irq = JACK_DETECT_INT; 也就是PM8921的第38Pin
mbhc_cfg.gpio_level_insert=1;表示headset插入时,中断JACK_DETECT_INT的值为高
如果不用GPIO中断检测耳机插入,则需要用到 TABLA_IRQ_MBHC_INSERTION 的内部中断
wcd9xxx_enable_irq(codec->control_data, TABLA_IRQ_MBHC_INSERTION);
3. tabla_mbhc_init_and_calibrate@wcd9310.c中设置中断回调函数tabla_mechanical_plug_detect_irq --> tabla_hs_gpio_handler
insert = (gpio_get_value_cansleep(tabla->mbhc_cfg.gpio) == tabla->mbhc_cfg.gpio_level_insert);
4. tabla_codec_detect_plug_type-->tabla_codec_get_plug_type@wcd9310.c 检测耳机类型
其中plug_type_ptr->v_no_mic =73,
plug_type_ptr->v_hs_max = 2850
重要结构static struct tabla_mbhc_config mbhc_cfg @kernel/sound/soc/msm/msm8960.c
mbhc_cfg.gpio_level_insert = 1 //表示GPIO38为高时为耳机插入
struct viskan_mbhc_data viskan_mbhc_data@kernel/arch/arm/mach-msm
.v_hs_max = 2850, //
S(v_no_mic, 73);
5. 支持的耳机jack顺序, MIC|GND|HPHR|HPHL, 在tabla_codec_get_plug_type@wcd9310.c中一共要读取4次电压
1和2时,分别读取两次MIC和GND之间的电压
3时,HPHR和GND都切换,读取两次MIC和GND之间的电压
4时,只有MIC切换,读取两次MIC和GND之间的电压,同时会判断是否为PLUG_TYPE_GND_MIC_SWAP或PLUG_TYPE_INVALID,没搞清楚具体硬件原理
如果 所有4次得到的MIC和GND之间的电压,均
< v_no_mic 则是headphone (PLUG_TYPE_HEADPHONE)
> v_hs_max 则是High impedance plug type,高阻态的耳机类型,线性输出(PLUG_TYPE_HIGH_HPH)
> v_no_mic & < v_hs_max,则是headset (PLUG_TYPE_HEADSET)
如果前后两次得到的耳机类型不一样,则说明该耳机不支持(PLUG_TYPE_INVALID)
支持的headset, 主要比较的是第一个电压mb_v[i]
DCE #1, 003e, V 1638, scaled V 1638, GND 0, VDDIO 0, inval 0
DCE #2, 0071, V 1733, scaled V 1733, GND 0, VDDIO 0, inval 0
DCE #3, ffa8, V 1360, scaled V 2092, GND 1, VDDIO 1, inval 1
DCE #4, ffa8, V 1360, scaled V 2092, GND 0, VDDIO 1, inval 0
不支持的headset
DCE #1, fe30, V 661, scaled V 661, GND 0, VDDIO 0, inval 0
DCE #2, fe30, V 661, scaled V 661, GND 0, VDDIO 0, inval 0
DCE #3, fd2f, V 183, scaled V 281, GND 1, VDDIO 1, inval 1
DCE #4, fe17, V 615, scaled V 946, GND 0, VDDIO 1, inval 1
支持的headphone
DCE #1, fcd2, V 11, scaled V 11, GND 0, VDDIO 0, inval 0
DCE #2, fcd2, V 11, scaled V 11, GND 0, VDDIO 0, inval 0
DCE #3, fcd0, V 7, scaled V 10, GND 1, VDDIO 1, inval 0
DCE #4, fcd0, V 7, scaled V 10, GND 0, VDDIO 1, inval 0
6. 根据测量audio jack中GND,MIC 脚与地之间的电压,来比较得出不同的耳机类型。目前只支持MIC|GND|HPHR|HPHL(CTIA)不支持(OMTP)
7. 耳机插入如何上报event
tabla_codec_detect_plug_type
-->tabla_codec_report_plug (只有PLUG_TYPE_GND_MIC_SWAP,PLUG_TYPE_HEADPHONE,PLUG_TYPE_HEADSET,PLUG_TYPE_HIGH_HPH才上报)
-->tabla_snd_soc_jack_report
-->snd_soc_jack_report_no_dapm
-->snd_jack_report@kernel/sound/core/jack.c
-->input_report_switch@kernel/include/linux/input.h
-->input_event(dev, EV_SW, code, !!value);
二十九,如何检测耳机的按键事件
1. tabla_hs_gpio_handler-->tabla_codec_detect_plug_type
只有耳机类型为headset 才能触发tabla_codec_start_hs_polling, 设置各按键的 电压门限值;(mbhc_state == MBHC_STATE_POTENTIAL)
2. 设置中断处理函数 @tabla_codec_probe
wcd9xxx_request_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL,tabla_dce_handler, "DC Estimation detect", tabla);
其中 tabla_hs_detect->tabla_mbhc_init_and_calibrate-->tabla_mbhc_cal会enable 中断TABLA_IRQ_MBHC_POTENTIAL
wcd9xxx_enable_irq(codec->control_data, TABLA_IRQ_MBHC_POTENTIAL);
3. 当headset插入后,如果有按键则会触发 TABLA_IRQ_MBHC_POTENTIAL 中断,从而call tabla_dce_handler-->tabla_determine_button
tabla_determine_button函数会通过读取的此时 micphone pin上的电压来决定是那个button key.
4. 而具体的按键是通过
@kernel/sound/soc/msm/msm8960.c中的函数snd_soc_jack_new(codec, "Button Jack",TABLA_JACK_BUTTON_MASK, &button_jack);注册一个jack
-->snd_jack_new
最终button_jack会接收按键信息
5. 根据测量micphone pin上的电压决定是那个按键,
而各按键的电压定义在 def_tabla_mbhc_cal@kernel/sound/soc/msm/msm8960.c
各个按键的结构定义为 struct tabla_mbhc_btn_detect_cfg@kernel/sound/soc/codecs/wcd9310.c
目前定义了4个按键 TABLA_MBHC_DEF_BUTTONS = 4,电压区间如下
btn_low[0] = -30;
btn_high[0] = 73;
btn_low[1] = 74;
btn_high[1] = 336;
btn_low[2] = 337;
btn_high[2] = 680;
btn_low[3] = 681;
btn_high[3] = 1257;
6. 在tabla_dce_handler函数中得到 具体是那个button按键被按下,然后检测中断TABLA_IRQ_MBHC_RELEASE (该中断一直是enable),当按键释放后触发tabla_release_handler-->tabla_snd_soc_jack_report 然后上报该event
-->snd_jack_report-->input_report_key@kernel/sound/core/jack.c
-->input_event(dev, EV_KEY, code, !!value);