wcd-mbhc-adc.c 35 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /* Copyright (c) 2017-2021, The Linux Foundation. All rights reserved.
  3. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
  4. */
  5. #include <linux/module.h>
  6. #include <linux/init.h>
  7. #include <linux/slab.h>
  8. #include <linux/of_gpio.h>
  9. #include <linux/platform_device.h>
  10. #include <linux/device.h>
  11. #include <linux/printk.h>
  12. #include <linux/ratelimit.h>
  13. #include <linux/list.h>
  14. #include <linux/bitops.h>
  15. #include <linux/delay.h>
  16. #include <linux/pm_runtime.h>
  17. #include <linux/kernel.h>
  18. #include <linux/input.h>
  19. #include <linux/firmware.h>
  20. #include <linux/completion.h>
  21. #include <sound/soc.h>
  22. #include <sound/jack.h>
  23. #include "wcd-mbhc-adc.h"
  24. #include <asoc/wcd-mbhc-v2.h>
  25. #include <asoc/pdata.h>
  26. #define WCD_MBHC_ADC_HS_THRESHOLD_MV 1700
  27. #define WCD_MBHC_ADC_HPH_THRESHOLD_MV 75
  28. #define WCD_MBHC_ADC_MICBIAS_MV 1800
  29. #define WCD_MBHC_FAKE_INS_RETRY 4
  30. static int wcd_mbhc_get_micbias(struct wcd_mbhc *mbhc)
  31. {
  32. int micbias = 0;
  33. u8 vout_ctl = 0;
  34. if (mbhc->mbhc_cb->get_micbias_val) {
  35. mbhc->mbhc_cb->get_micbias_val(mbhc, &micbias);
  36. pr_debug("%s: micbias: %d\n", __func__, micbias);
  37. } else {
  38. /* Read MBHC Micbias (Mic Bias2) voltage */
  39. WCD_MBHC_REG_READ(WCD_MBHC_MICB2_VOUT, vout_ctl);
  40. /* Formula for getting micbias from vout
  41. * micbias = 1.0V + VOUT_CTL * 50mV
  42. */
  43. micbias = 1000 + (vout_ctl * 50);
  44. pr_debug("%s: vout_ctl: %d, micbias: %d\n",
  45. __func__, vout_ctl, micbias);
  46. }
  47. return micbias;
  48. }
  49. static int wcd_get_voltage_from_adc(u8 val, int micbias)
  50. {
  51. /* Formula for calculating voltage from ADC
  52. * Voltage = ADC_RESULT*12.5mV*V_MICBIAS/1.8
  53. */
  54. return ((val * 125 * micbias)/(WCD_MBHC_ADC_MICBIAS_MV * 10));
  55. }
  56. static int wcd_measure_adc_continuous(struct wcd_mbhc *mbhc)
  57. {
  58. u8 adc_result = 0;
  59. int output_mv = 0;
  60. int retry = 3;
  61. u8 adc_en = 0;
  62. pr_debug("%s: enter\n", __func__);
  63. /* Pre-requisites for ADC continuous measurement */
  64. /* Read legacy electircal detection and disable */
  65. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_SCHMT_ISRC, 0x00);
  66. /* Set ADC to continuous measurement */
  67. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, 1);
  68. /* Read ADC Enable bit to restore after adc measurement */
  69. WCD_MBHC_REG_READ(WCD_MBHC_ADC_EN, adc_en);
  70. /* Disable ADC_ENABLE bit */
  71. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 0);
  72. /* Disable MBHC FSM */
  73. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 0);
  74. /* Set the MUX selection to IN2P */
  75. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_MUX_CTL, MUX_CTL_IN2P);
  76. /*
  77. * Current source mode requires Auto zeroing to be enabled
  78. * automatically. If HW doesn't do it, SW has to take care of this
  79. * for button interrupts to work fine and to avoid
  80. * fake electrical removal interrupts by enabling autozero before FSM
  81. * enable and disable it after FSM enable
  82. */
  83. if (mbhc->mbhc_cb->mbhc_comp_autozero_control)
  84. mbhc->mbhc_cb->mbhc_comp_autozero_control(mbhc,
  85. true);
  86. /* Enable MBHC FSM */
  87. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 1);
  88. if (mbhc->mbhc_cb->mbhc_comp_autozero_control)
  89. mbhc->mbhc_cb->mbhc_comp_autozero_control(mbhc,
  90. false);
  91. /* Enable ADC_ENABLE bit */
  92. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 1);
  93. while (retry--) {
  94. /* wait for 3 msec before reading ADC result */
  95. usleep_range(3000, 3100);
  96. /* Read ADC result */
  97. WCD_MBHC_REG_READ(WCD_MBHC_ADC_RESULT, adc_result);
  98. }
  99. /* Restore ADC Enable */
  100. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, adc_en);
  101. /* Get voltage from ADC result */
  102. output_mv = wcd_get_voltage_from_adc(adc_result,
  103. wcd_mbhc_get_micbias(mbhc));
  104. pr_debug("%s: adc_result: 0x%x, output_mv: %d\n",
  105. __func__, adc_result, output_mv);
  106. return output_mv;
  107. }
  108. static int wcd_measure_adc_once(struct wcd_mbhc *mbhc, int mux_ctl)
  109. {
  110. u8 adc_timeout = 0;
  111. u8 adc_complete = 0;
  112. u8 adc_result = 0;
  113. int retry = 6;
  114. int ret = 0;
  115. int output_mv = 0;
  116. u8 adc_en = 0;
  117. bool is_pa_on = false;
  118. pr_debug("%s: enter\n", __func__);
  119. if (mbhc->mbhc_cb->hph_pa_on_status) {
  120. if (mbhc->mbhc_cb->hph_pa_on_status(mbhc->component)) {
  121. mbhc->mbhc_cb->hph_pa_enable(mbhc->component, 0);
  122. is_pa_on = true;
  123. pr_debug("%s: pa is on before detection,so disable pa and read adc \n", __func__);
  124. }
  125. }
  126. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, 0);
  127. /* Read ADC Enable bit to restore after adc measurement */
  128. WCD_MBHC_REG_READ(WCD_MBHC_ADC_EN, adc_en);
  129. /* Trigger ADC one time measurement */
  130. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 0);
  131. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 0);
  132. /* Set the appropriate MUX selection */
  133. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_MUX_CTL, mux_ctl);
  134. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 1);
  135. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 1);
  136. while (retry--) {
  137. /* wait for 600usec to get adc results */
  138. usleep_range(600, 610);
  139. /* check for ADC Timeout */
  140. WCD_MBHC_REG_READ(WCD_MBHC_ADC_TIMEOUT, adc_timeout);
  141. if (adc_timeout)
  142. continue;
  143. /* Read ADC complete bit */
  144. WCD_MBHC_REG_READ(WCD_MBHC_ADC_COMPLETE, adc_complete);
  145. if (!adc_complete)
  146. continue;
  147. /* Read ADC result */
  148. WCD_MBHC_REG_READ(WCD_MBHC_ADC_RESULT, adc_result);
  149. pr_debug("%s: ADC result: 0x%x\n", __func__, adc_result);
  150. /* Get voltage from ADC result */
  151. output_mv = wcd_get_voltage_from_adc(adc_result,
  152. wcd_mbhc_get_micbias(mbhc));
  153. break;
  154. }
  155. /* Restore ADC Enable */
  156. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, adc_en);
  157. if (retry <= 0) {
  158. pr_err("%s: adc complete: %d, adc timeout: %d\n",
  159. __func__, adc_complete, adc_timeout);
  160. ret = -EINVAL;
  161. } else {
  162. pr_debug("%s: adc complete: %d, adc timeout: %d output_mV: %d\n",
  163. __func__, adc_complete, adc_timeout, output_mv);
  164. ret = output_mv;
  165. }
  166. if (is_pa_on) {
  167. mbhc->mbhc_cb->hph_pa_enable(mbhc->component, 1);
  168. pr_debug("%s: restore pa \n", __func__);
  169. }
  170. pr_debug("%s: leave\n", __func__);
  171. return ret;
  172. }
  173. static bool wcd_mbhc_adc_detect_anc_plug_type(struct wcd_mbhc *mbhc)
  174. {
  175. bool anc_mic_found = false;
  176. u16 fsm_en = 0;
  177. u8 det = 0;
  178. unsigned long retry = 0;
  179. int valid_plug_cnt = 0, invalid_plug_cnt = 0;
  180. int ret = 0;
  181. u8 elect_ctl = 0;
  182. u8 adc_mode = 0;
  183. u8 vref = 0;
  184. int vref_mv[] = {1650, 1500, 1600, 1700};
  185. if (mbhc->mbhc_cfg->anc_micbias < MIC_BIAS_1 ||
  186. mbhc->mbhc_cfg->anc_micbias > MIC_BIAS_4)
  187. return false;
  188. if (!mbhc->mbhc_cb->mbhc_micbias_control)
  189. return false;
  190. /* Disable Detection done for ADC operation */
  191. WCD_MBHC_REG_READ(WCD_MBHC_DETECTION_DONE, det);
  192. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 0);
  193. /* Mask ADC COMPLETE interrupt */
  194. wcd_mbhc_hs_elec_irq(mbhc, WCD_MBHC_ELEC_HS_INS, false);
  195. WCD_MBHC_REG_READ(WCD_MBHC_FSM_EN, fsm_en);
  196. mbhc->mbhc_cb->mbhc_micbias_control(mbhc->component,
  197. mbhc->mbhc_cfg->anc_micbias,
  198. MICB_ENABLE);
  199. /* Read legacy electircal detection and disable */
  200. WCD_MBHC_REG_READ(WCD_MBHC_ELECT_SCHMT_ISRC, elect_ctl);
  201. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_SCHMT_ISRC, 0x00);
  202. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ANC_DET_EN, 1);
  203. WCD_MBHC_REG_READ(WCD_MBHC_ADC_MODE, adc_mode);
  204. /*
  205. * wait for button debounce time 20ms. If 4-pole plug is inserted
  206. * into 5-pole jack, then there will be a button press interrupt
  207. * during anc plug detection. In that case though Hs_comp_res is 0,
  208. * it should not be declared as ANC plug type
  209. */
  210. usleep_range(20000, 20100);
  211. /*
  212. * After enabling FSM, to handle slow insertion scenarios,
  213. * check IN3 voltage is below the Vref
  214. */
  215. WCD_MBHC_REG_READ(WCD_MBHC_HS_VREF, vref);
  216. do {
  217. if (wcd_swch_level_remove(mbhc)) {
  218. pr_debug("%s: Switch level is low\n", __func__);
  219. goto done;
  220. }
  221. pr_debug("%s: Retry attempt %lu\n", __func__, retry + 1);
  222. ret = wcd_measure_adc_once(mbhc, MUX_CTL_IN3P);
  223. /* TODO - check the logic */
  224. if (ret && (ret < vref_mv[vref]))
  225. valid_plug_cnt++;
  226. else
  227. invalid_plug_cnt++;
  228. retry++;
  229. } while (retry < ANC_DETECT_RETRY_CNT);
  230. pr_debug("%s: valid: %d, invalid: %d\n", __func__, valid_plug_cnt,
  231. invalid_plug_cnt);
  232. /* decision logic */
  233. if (valid_plug_cnt > invalid_plug_cnt)
  234. anc_mic_found = true;
  235. done:
  236. /* Restore ADC mode */
  237. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, adc_mode);
  238. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ANC_DET_EN, 0);
  239. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 0);
  240. /* Set the MUX selection to AUTO */
  241. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_MUX_CTL, MUX_CTL_AUTO);
  242. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 1);
  243. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, fsm_en);
  244. /* Restore detection done */
  245. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, det);
  246. /* Restore electrical detection */
  247. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_SCHMT_ISRC, elect_ctl);
  248. mbhc->mbhc_cb->mbhc_micbias_control(mbhc->component,
  249. mbhc->mbhc_cfg->anc_micbias,
  250. MICB_DISABLE);
  251. pr_debug("%s: anc mic %sfound\n", __func__,
  252. anc_mic_found ? "" : "not ");
  253. return anc_mic_found;
  254. }
  255. /* To determine if cross connection occurred */
  256. static int wcd_check_cross_conn(struct wcd_mbhc *mbhc)
  257. {
  258. enum wcd_mbhc_plug_type plug_type = MBHC_PLUG_TYPE_NONE;
  259. int hphl_adc_res = 0, hphr_adc_res = 0;
  260. u8 fsm_en = 0;
  261. int ret = 0;
  262. u8 adc_mode = 0;
  263. u8 elect_ctl = 0;
  264. u8 adc_en = 0;
  265. pr_debug("%s: enter\n", __func__);
  266. /* Check for button press and plug detection */
  267. if (wcd_swch_level_remove(mbhc)) {
  268. pr_debug("%s: Switch level is low\n", __func__);
  269. return -EINVAL;
  270. }
  271. /* If PA is enabled, dont check for cross-connection */
  272. if (mbhc->mbhc_cb->hph_pa_on_status)
  273. if (mbhc->mbhc_cb->hph_pa_on_status(mbhc->component))
  274. return -EINVAL;
  275. /* Read legacy electircal detection and disable */
  276. WCD_MBHC_REG_READ(WCD_MBHC_ELECT_SCHMT_ISRC, elect_ctl);
  277. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_SCHMT_ISRC, 0x00);
  278. /* Disable surge detection before ADC measurement */
  279. if (mbhc->mbhc_cb->mbhc_surge_ctl)
  280. mbhc->mbhc_cb->mbhc_surge_ctl(mbhc, false);
  281. /* Read and set ADC to single measurement */
  282. WCD_MBHC_REG_READ(WCD_MBHC_ADC_MODE, adc_mode);
  283. /* Read ADC Enable bit to restore after adc measurement */
  284. WCD_MBHC_REG_READ(WCD_MBHC_ADC_EN, adc_en);
  285. /* Read FSM status */
  286. WCD_MBHC_REG_READ(WCD_MBHC_FSM_EN, fsm_en);
  287. /* Get adc result for HPH L */
  288. hphl_adc_res = wcd_measure_adc_once(mbhc, MUX_CTL_HPH_L);
  289. if (hphl_adc_res < 0) {
  290. pr_err("%s: hphl_adc_res adc measurement failed\n", __func__);
  291. ret = hphl_adc_res;
  292. goto done;
  293. }
  294. /* Get adc result for HPH R in mV */
  295. hphr_adc_res = wcd_measure_adc_once(mbhc, MUX_CTL_HPH_R);
  296. if (hphr_adc_res < 0) {
  297. pr_err("%s: hphr_adc_res adc measurement failed\n", __func__);
  298. ret = hphr_adc_res;
  299. goto done;
  300. }
  301. /* Update cross connection threshold voltages if needed */
  302. if (mbhc->mbhc_cb->update_cross_conn_thr)
  303. mbhc->mbhc_cb->update_cross_conn_thr(mbhc);
  304. if (hphl_adc_res > mbhc->hphl_cross_conn_thr ||
  305. hphr_adc_res > mbhc->hphr_cross_conn_thr) {
  306. plug_type = MBHC_PLUG_TYPE_GND_MIC_SWAP;
  307. pr_debug("%s: Cross connection identified\n", __func__);
  308. } else {
  309. pr_debug("%s: No Cross connection found\n", __func__);
  310. }
  311. done:
  312. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 0);
  313. /* Set the MUX selection to Auto */
  314. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_MUX_CTL, MUX_CTL_AUTO);
  315. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 1);
  316. /* Restore ADC Enable */
  317. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, adc_en);
  318. /* Restore ADC mode */
  319. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, adc_mode);
  320. /* Restore FSM state */
  321. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, fsm_en);
  322. /* Restore surge detection */
  323. if (mbhc->mbhc_cb->mbhc_surge_ctl)
  324. mbhc->mbhc_cb->mbhc_surge_ctl(mbhc, true);
  325. /* Restore electrical detection */
  326. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_SCHMT_ISRC, elect_ctl);
  327. pr_debug("%s: leave, plug type: %d\n", __func__, plug_type);
  328. return (plug_type == MBHC_PLUG_TYPE_GND_MIC_SWAP) ? true : false;
  329. }
  330. static int wcd_mbhc_adc_get_spl_hs_thres(struct wcd_mbhc *mbhc)
  331. {
  332. int hs_threshold, micbias_mv;
  333. micbias_mv = wcd_mbhc_get_micbias(mbhc);
  334. if (mbhc->hs_thr && mbhc->micb_mv != WCD_MBHC_ADC_MICBIAS_MV) {
  335. if (mbhc->micb_mv == micbias_mv)
  336. hs_threshold = mbhc->hs_thr;
  337. else
  338. hs_threshold = (mbhc->hs_thr *
  339. micbias_mv) / mbhc->micb_mv;
  340. } else {
  341. hs_threshold = ((WCD_MBHC_ADC_HS_THRESHOLD_MV *
  342. micbias_mv) / WCD_MBHC_ADC_MICBIAS_MV);
  343. }
  344. return hs_threshold;
  345. }
  346. static int wcd_mbhc_adc_get_hs_thres(struct wcd_mbhc *mbhc)
  347. {
  348. int hs_threshold, micbias_mv;
  349. micbias_mv = wcd_mbhc_get_micbias(mbhc);
  350. if (mbhc->hs_thr) {
  351. if (mbhc->micb_mv == micbias_mv)
  352. hs_threshold = mbhc->hs_thr;
  353. else
  354. hs_threshold = (mbhc->hs_thr *
  355. micbias_mv) / mbhc->micb_mv;
  356. } else {
  357. hs_threshold = ((WCD_MBHC_ADC_HS_THRESHOLD_MV *
  358. micbias_mv) / WCD_MBHC_ADC_MICBIAS_MV);
  359. }
  360. return hs_threshold;
  361. }
  362. static int wcd_mbhc_adc_get_hph_thres(struct wcd_mbhc *mbhc)
  363. {
  364. int hph_threshold, micbias_mv;
  365. micbias_mv = wcd_mbhc_get_micbias(mbhc);
  366. if (mbhc->hph_thr) {
  367. if (mbhc->micb_mv == micbias_mv)
  368. hph_threshold = mbhc->hph_thr;
  369. else
  370. hph_threshold = (mbhc->hph_thr *
  371. micbias_mv) / mbhc->micb_mv;
  372. } else {
  373. hph_threshold = ((WCD_MBHC_ADC_HPH_THRESHOLD_MV *
  374. micbias_mv) / WCD_MBHC_ADC_MICBIAS_MV);
  375. }
  376. return hph_threshold;
  377. }
  378. static bool wcd_mbhc_adc_check_for_spl_headset(struct wcd_mbhc *mbhc,
  379. int *spl_hs_cnt)
  380. {
  381. bool spl_hs = false;
  382. int output_mv = 0;
  383. int adc_threshold = 0, adc_hph_threshold = 0;
  384. pr_debug("%s: enter\n", __func__);
  385. if (!mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic)
  386. goto exit;
  387. /* Bump up MB2 to 2.7V */
  388. mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic(mbhc->component,
  389. mbhc->mbhc_cfg->mbhc_micbias, true);
  390. usleep_range(10000, 10100);
  391. /*
  392. * Use ADC single mode to minimize the chance of missing out
  393. * btn press/relesae for HEADSET type during correct work.
  394. */
  395. output_mv = wcd_measure_adc_once(mbhc, MUX_CTL_IN2P);
  396. adc_threshold = wcd_mbhc_adc_get_spl_hs_thres(mbhc);
  397. adc_hph_threshold = wcd_mbhc_adc_get_hph_thres(mbhc);
  398. if (output_mv > adc_threshold || output_mv < adc_hph_threshold) {
  399. spl_hs = false;
  400. } else {
  401. spl_hs = true;
  402. if (spl_hs_cnt)
  403. *spl_hs_cnt += 1;
  404. }
  405. /* MB2 back to 1.8v if the type is not special headset */
  406. if (spl_hs_cnt && (*spl_hs_cnt != WCD_MBHC_SPL_HS_CNT)) {
  407. mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic(mbhc->component,
  408. mbhc->mbhc_cfg->mbhc_micbias, false);
  409. /* Add 10ms delay for micbias to settle */
  410. usleep_range(10000, 10100);
  411. }
  412. if (spl_hs)
  413. pr_debug("%s: Detected special HS (%d)\n", __func__, spl_hs);
  414. exit:
  415. pr_debug("%s: leave\n", __func__);
  416. return spl_hs;
  417. }
  418. static bool wcd_is_special_headset(struct wcd_mbhc *mbhc)
  419. {
  420. int delay = 0;
  421. bool ret = false;
  422. bool is_spl_hs = false;
  423. int output_mv = 0;
  424. int adc_threshold = 0;
  425. /*
  426. * Increase micbias to 2.7V to detect headsets with
  427. * threshold on microphone
  428. */
  429. if (mbhc->mbhc_cb->mbhc_micbias_control &&
  430. !mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic) {
  431. pr_debug("%s: callback fn micb_ctrl_thr_mic not defined\n",
  432. __func__);
  433. return false;
  434. } else if (mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic) {
  435. ret = mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic(mbhc->component,
  436. MIC_BIAS_2, true);
  437. if (ret) {
  438. pr_err("%s: mbhc_micb_ctrl_thr_mic failed, ret: %d\n",
  439. __func__, ret);
  440. return false;
  441. }
  442. }
  443. adc_threshold = wcd_mbhc_adc_get_spl_hs_thres(mbhc);
  444. while (!is_spl_hs) {
  445. if (mbhc->hs_detect_work_stop) {
  446. pr_debug("%s: stop requested: %d\n", __func__,
  447. mbhc->hs_detect_work_stop);
  448. break;
  449. }
  450. delay += 50;
  451. /* Wait for 50ms for FSM to update result */
  452. msleep(50);
  453. output_mv = wcd_measure_adc_once(mbhc, MUX_CTL_IN2P);
  454. if (output_mv <= adc_threshold) {
  455. pr_debug("%s: Special headset detected in %d msecs\n",
  456. __func__, delay);
  457. is_spl_hs = true;
  458. }
  459. if (delay == SPECIAL_HS_DETECT_TIME_MS) {
  460. pr_debug("%s: Spl headset not found in 2 sec\n",
  461. __func__);
  462. break;
  463. }
  464. }
  465. if (is_spl_hs) {
  466. pr_debug("%s: Headset with threshold found\n", __func__);
  467. mbhc->micbias_enable = true;
  468. ret = true;
  469. }
  470. if (mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic &&
  471. !mbhc->micbias_enable)
  472. mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic(mbhc->component,
  473. MIC_BIAS_2,
  474. false);
  475. pr_debug("%s: leave, micb_enable: %d\n", __func__,
  476. mbhc->micbias_enable);
  477. return ret;
  478. }
  479. static void wcd_mbhc_adc_update_fsm_source(struct wcd_mbhc *mbhc,
  480. enum wcd_mbhc_plug_type plug_type)
  481. {
  482. bool micbias2;
  483. micbias2 = mbhc->mbhc_cb->micbias_enable_status(mbhc,
  484. MIC_BIAS_2);
  485. switch (plug_type) {
  486. case MBHC_PLUG_TYPE_HEADPHONE:
  487. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_BTN_ISRC_CTL, 3);
  488. break;
  489. case MBHC_PLUG_TYPE_HEADSET:
  490. case MBHC_PLUG_TYPE_ANC_HEADPHONE:
  491. if (!mbhc->is_hs_recording && !micbias2)
  492. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_BTN_ISRC_CTL, 3);
  493. break;
  494. default:
  495. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_BTN_ISRC_CTL, 0);
  496. break;
  497. };
  498. }
  499. /* should be called under interrupt context that hold suspend */
  500. static void wcd_schedule_hs_detect_plug(struct wcd_mbhc *mbhc,
  501. struct work_struct *work)
  502. {
  503. pr_debug("%s: scheduling correct_swch_plug\n", __func__);
  504. WCD_MBHC_RSC_ASSERT_LOCKED(mbhc);
  505. mbhc->hs_detect_work_stop = false;
  506. mbhc->mbhc_cb->lock_sleep(mbhc, true);
  507. schedule_work(work);
  508. }
  509. /* called under codec_resource_lock acquisition */
  510. static void wcd_cancel_hs_detect_plug(struct wcd_mbhc *mbhc,
  511. struct work_struct *work)
  512. {
  513. pr_debug("%s: Canceling correct_plug_swch\n", __func__);
  514. mbhc->hs_detect_work_stop = true;
  515. WCD_MBHC_RSC_UNLOCK(mbhc);
  516. if (cancel_work_sync(work)) {
  517. pr_debug("%s: correct_plug_swch is canceled\n",
  518. __func__);
  519. mbhc->mbhc_cb->lock_sleep(mbhc, false);
  520. }
  521. WCD_MBHC_RSC_LOCK(mbhc);
  522. }
  523. /* called under codec_resource_lock acquisition */
  524. static void wcd_mbhc_adc_detect_plug_type(struct wcd_mbhc *mbhc)
  525. {
  526. struct snd_soc_component *component = mbhc->component;
  527. pr_debug("%s: enter\n", __func__);
  528. WCD_MBHC_RSC_ASSERT_LOCKED(mbhc);
  529. if (mbhc->mbhc_cb->hph_pull_down_ctrl)
  530. mbhc->mbhc_cb->hph_pull_down_ctrl(component, false);
  531. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 0);
  532. if (mbhc->mbhc_cb->mbhc_micbias_control) {
  533. mbhc->mbhc_cb->mbhc_micbias_control(component, MIC_BIAS_2,
  534. MICB_ENABLE);
  535. } else {
  536. pr_err("%s: Mic Bias is not enabled\n", __func__);
  537. return;
  538. }
  539. /* Re-initialize button press completion object */
  540. reinit_completion(&mbhc->btn_press_compl);
  541. wcd_schedule_hs_detect_plug(mbhc, &mbhc->correct_plug_swch);
  542. pr_debug("%s: leave\n", __func__);
  543. }
  544. static void wcd_micbias_disable(struct wcd_mbhc *mbhc)
  545. {
  546. if (mbhc->micbias_enable) {
  547. mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic(
  548. mbhc->component, MIC_BIAS_2, false);
  549. if (mbhc->mbhc_cb->set_micbias_value)
  550. mbhc->mbhc_cb->set_micbias_value(
  551. mbhc->component);
  552. mbhc->micbias_enable = false;
  553. }
  554. }
  555. static int wcd_mbhc_get_plug_from_adc(struct wcd_mbhc *mbhc, int adc_result)
  556. {
  557. enum wcd_mbhc_plug_type plug_type = MBHC_PLUG_TYPE_INVALID;
  558. u32 hph_thr = 0, hs_thr = 0;
  559. hs_thr = wcd_mbhc_adc_get_hs_thres(mbhc);
  560. hph_thr = wcd_mbhc_adc_get_hph_thres(mbhc);
  561. if (adc_result < hph_thr)
  562. plug_type = MBHC_PLUG_TYPE_HEADPHONE;
  563. else if (adc_result > hs_thr)
  564. plug_type = MBHC_PLUG_TYPE_HIGH_HPH;
  565. else
  566. plug_type = MBHC_PLUG_TYPE_HEADSET;
  567. pr_debug("%s: plug type is %d found\n", __func__, plug_type);
  568. return plug_type;
  569. }
  570. static void wcd_correct_swch_plug(struct work_struct *work)
  571. {
  572. struct wcd_mbhc *mbhc;
  573. struct snd_soc_component *component;
  574. enum wcd_mbhc_plug_type plug_type = MBHC_PLUG_TYPE_INVALID;
  575. unsigned long timeout;
  576. bool wrk_complete = false;
  577. int pt_gnd_mic_swap_cnt = 0;
  578. int no_gnd_mic_swap_cnt = 0;
  579. bool is_pa_on = false, spl_hs = false, spl_hs_reported = false;
  580. int ret = 0;
  581. int spl_hs_count = 0;
  582. int output_mv = 0;
  583. int cross_conn;
  584. int try = 0;
  585. int hs_threshold, micbias_mv;
  586. pr_debug("%s: enter\n", __func__);
  587. mbhc = container_of(work, struct wcd_mbhc, correct_plug_swch);
  588. component = mbhc->component;
  589. micbias_mv = wcd_mbhc_get_micbias(mbhc);
  590. hs_threshold = wcd_mbhc_adc_get_hs_thres(mbhc);
  591. WCD_MBHC_RSC_LOCK(mbhc);
  592. /* Mask ADC COMPLETE interrupt */
  593. wcd_mbhc_hs_elec_irq(mbhc, WCD_MBHC_ELEC_HS_INS, false);
  594. WCD_MBHC_RSC_UNLOCK(mbhc);
  595. /* Check for cross connection */
  596. do {
  597. cross_conn = wcd_check_cross_conn(mbhc);
  598. try++;
  599. } while (try < mbhc->swap_thr);
  600. if (cross_conn > 0) {
  601. plug_type = MBHC_PLUG_TYPE_GND_MIC_SWAP;
  602. pr_debug("%s: cross connection found, Plug type %d\n",
  603. __func__, plug_type);
  604. goto correct_plug_type;
  605. }
  606. /* Find plug type */
  607. output_mv = wcd_measure_adc_continuous(mbhc);
  608. plug_type = wcd_mbhc_get_plug_from_adc(mbhc, output_mv);
  609. /*
  610. * Report plug type if it is either headset or headphone
  611. * else start the 3 sec loop
  612. */
  613. if ((plug_type == MBHC_PLUG_TYPE_HEADSET ||
  614. plug_type == MBHC_PLUG_TYPE_HEADPHONE) &&
  615. (!wcd_swch_level_remove(mbhc))) {
  616. WCD_MBHC_RSC_LOCK(mbhc);
  617. wcd_mbhc_find_plug_and_report(mbhc, plug_type);
  618. WCD_MBHC_RSC_UNLOCK(mbhc);
  619. }
  620. /*
  621. * Set DETECTION_DONE bit for HEADSET and ANC_HEADPHONE,
  622. * so that btn press/release interrupt can be generated.
  623. */
  624. if (mbhc->current_plug == MBHC_PLUG_TYPE_HEADSET ||
  625. mbhc->current_plug == MBHC_PLUG_TYPE_ANC_HEADPHONE) {
  626. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, 0);
  627. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 0);
  628. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 1);
  629. }
  630. correct_plug_type:
  631. /*
  632. * Callback to disable BCS slow insertion detection
  633. */
  634. if (mbhc->mbhc_cb->bcs_enable)
  635. mbhc->mbhc_cb->bcs_enable(mbhc, false);
  636. timeout = jiffies + msecs_to_jiffies(HS_DETECT_PLUG_TIME_MS);
  637. while (!time_after(jiffies, timeout)) {
  638. if (mbhc->hs_detect_work_stop) {
  639. pr_debug("%s: stop requested: %d\n", __func__,
  640. mbhc->hs_detect_work_stop);
  641. wcd_micbias_disable(mbhc);
  642. goto exit;
  643. }
  644. /* allow sometime and re-check stop requested again */
  645. msleep(20);
  646. if (mbhc->hs_detect_work_stop) {
  647. pr_debug("%s: stop requested: %d\n", __func__,
  648. mbhc->hs_detect_work_stop);
  649. wcd_micbias_disable(mbhc);
  650. goto exit;
  651. }
  652. msleep(180);
  653. /*
  654. * Use ADC single mode to minimize the chance of missing out
  655. * btn press/release for HEADSET type during correct work.
  656. */
  657. output_mv = wcd_measure_adc_once(mbhc, MUX_CTL_IN2P);
  658. /*
  659. * instead of hogging system by contineous polling, wait for
  660. * sometime and re-check stop request again.
  661. */
  662. plug_type = wcd_mbhc_get_plug_from_adc(mbhc, output_mv);
  663. if ((output_mv > hs_threshold) &&
  664. (spl_hs_count < WCD_MBHC_SPL_HS_CNT)) {
  665. spl_hs = wcd_mbhc_adc_check_for_spl_headset(mbhc,
  666. &spl_hs_count);
  667. output_mv = wcd_measure_adc_once(mbhc, MUX_CTL_IN2P);
  668. if (spl_hs_count == WCD_MBHC_SPL_HS_CNT) {
  669. hs_threshold = (hs_threshold *
  670. wcd_mbhc_get_micbias(mbhc)) / micbias_mv;
  671. spl_hs = true;
  672. mbhc->micbias_enable = true;
  673. }
  674. }
  675. if (mbhc->mbhc_cb->hph_pa_on_status)
  676. is_pa_on = mbhc->mbhc_cb->hph_pa_on_status(
  677. mbhc->component);
  678. if ((output_mv <= hs_threshold) &&
  679. (!is_pa_on)) {
  680. /* Check for cross connection*/
  681. ret = wcd_check_cross_conn(mbhc);
  682. if (ret < 0)
  683. continue;
  684. else if (ret > 0) {
  685. pt_gnd_mic_swap_cnt++;
  686. no_gnd_mic_swap_cnt = 0;
  687. if (pt_gnd_mic_swap_cnt <
  688. mbhc->swap_thr) {
  689. continue;
  690. } else if (pt_gnd_mic_swap_cnt >
  691. mbhc->swap_thr) {
  692. /*
  693. * This is due to GND/MIC switch didn't
  694. * work, Report unsupported plug.
  695. */
  696. pr_debug("%s: switch did not work\n",
  697. __func__);
  698. plug_type = MBHC_PLUG_TYPE_GND_MIC_SWAP;
  699. goto report;
  700. } else {
  701. plug_type = MBHC_PLUG_TYPE_GND_MIC_SWAP;
  702. }
  703. } else {
  704. no_gnd_mic_swap_cnt++;
  705. pt_gnd_mic_swap_cnt = 0;
  706. plug_type = wcd_mbhc_get_plug_from_adc(
  707. mbhc, output_mv);
  708. if ((no_gnd_mic_swap_cnt <
  709. mbhc->swap_thr) &&
  710. (spl_hs_count != WCD_MBHC_SPL_HS_CNT)) {
  711. continue;
  712. } else {
  713. no_gnd_mic_swap_cnt = 0;
  714. }
  715. }
  716. if ((pt_gnd_mic_swap_cnt == mbhc->swap_thr) &&
  717. (plug_type == MBHC_PLUG_TYPE_GND_MIC_SWAP)) {
  718. /*
  719. * if switch is toggled, check again,
  720. * otherwise report unsupported plug
  721. */
  722. if (mbhc->mbhc_cfg->swap_gnd_mic &&
  723. mbhc->mbhc_cfg->swap_gnd_mic(component,
  724. true)) {
  725. pr_debug("%s: US_EU gpio present,flip switch\n"
  726. , __func__);
  727. continue;
  728. }
  729. }
  730. }
  731. if (output_mv > hs_threshold) {
  732. pr_debug("%s: cable is extension cable\n", __func__);
  733. plug_type = MBHC_PLUG_TYPE_HIGH_HPH;
  734. wrk_complete = true;
  735. } else {
  736. pr_debug("%s: cable might be headset: %d\n", __func__,
  737. plug_type);
  738. if (plug_type != MBHC_PLUG_TYPE_GND_MIC_SWAP) {
  739. plug_type = wcd_mbhc_get_plug_from_adc(
  740. mbhc, output_mv);
  741. if (!spl_hs_reported &&
  742. spl_hs_count == WCD_MBHC_SPL_HS_CNT) {
  743. spl_hs_reported = true;
  744. WCD_MBHC_RSC_LOCK(mbhc);
  745. wcd_mbhc_find_plug_and_report(mbhc,
  746. plug_type);
  747. WCD_MBHC_RSC_UNLOCK(mbhc);
  748. continue;
  749. } else if (spl_hs_reported)
  750. continue;
  751. /*
  752. * Report headset only if not already reported
  753. * and if there is not button press without
  754. * release
  755. */
  756. if ((mbhc->current_plug !=
  757. MBHC_PLUG_TYPE_HEADSET) &&
  758. (mbhc->current_plug !=
  759. MBHC_PLUG_TYPE_ANC_HEADPHONE) &&
  760. !wcd_swch_level_remove(mbhc)) {
  761. pr_debug("%s: cable is %s headset\n",
  762. __func__,
  763. ((spl_hs_count ==
  764. WCD_MBHC_SPL_HS_CNT) ?
  765. "special ":""));
  766. goto report;
  767. }
  768. }
  769. wrk_complete = false;
  770. }
  771. }
  772. if (!wrk_complete) {
  773. /*
  774. * If plug_tye is headset, we might have already reported either
  775. * in detect_plug-type or in above while loop, no need to report
  776. * again
  777. */
  778. if ((plug_type == MBHC_PLUG_TYPE_HEADSET) ||
  779. (plug_type == MBHC_PLUG_TYPE_ANC_HEADPHONE)) {
  780. pr_debug("%s: plug_type:0x%x current_plug: 0x%x already reported\n",
  781. __func__, plug_type, mbhc->current_plug);
  782. if (mbhc->current_plug != plug_type)
  783. goto report;
  784. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, 0);
  785. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 0);
  786. goto enable_supply;
  787. }
  788. }
  789. if (plug_type == MBHC_PLUG_TYPE_HIGH_HPH) {
  790. if (wcd_is_special_headset(mbhc)) {
  791. pr_debug("%s: Special headset found %d\n",
  792. __func__, plug_type);
  793. plug_type = MBHC_PLUG_TYPE_HEADSET;
  794. } else {
  795. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_ISRC_EN, 1);
  796. }
  797. }
  798. report:
  799. if (wcd_swch_level_remove(mbhc)) {
  800. pr_debug("%s: Switch level is low\n", __func__);
  801. goto exit;
  802. }
  803. pr_debug("%s: Valid plug found, plug type %d wrk_cmpt %d btn_intr %d\n",
  804. __func__, plug_type, wrk_complete,
  805. mbhc->btn_press_intr);
  806. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, 0);
  807. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 0);
  808. WCD_MBHC_RSC_LOCK(mbhc);
  809. wcd_mbhc_find_plug_and_report(mbhc, plug_type);
  810. WCD_MBHC_RSC_UNLOCK(mbhc);
  811. enable_supply:
  812. /*
  813. * Set DETECTION_DONE bit for HEADSET and ANC_HEADPHONE,
  814. * so that btn press/release interrupt can be generated.
  815. * For other plug type, clear the bit.
  816. */
  817. if (plug_type == MBHC_PLUG_TYPE_HEADSET ||
  818. plug_type == MBHC_PLUG_TYPE_ANC_HEADPHONE)
  819. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 1);
  820. else
  821. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 0);
  822. if (mbhc->mbhc_cb->bcs_enable)
  823. mbhc->mbhc_cb->bcs_enable(mbhc, true);
  824. if (mbhc->mbhc_cb->mbhc_micbias_control)
  825. wcd_mbhc_adc_update_fsm_source(mbhc, plug_type);
  826. exit:
  827. if (mbhc->mbhc_cb->mbhc_micbias_control &&
  828. !mbhc->micbias_enable)
  829. mbhc->mbhc_cb->mbhc_micbias_control(component, MIC_BIAS_2,
  830. MICB_DISABLE);
  831. /*
  832. * If plug type is corrected from special headset to headphone,
  833. * clear the micbias enable flag, set micbias back to 1.8V and
  834. * disable micbias.
  835. */
  836. if (plug_type == MBHC_PLUG_TYPE_HEADPHONE &&
  837. mbhc->micbias_enable) {
  838. if (mbhc->mbhc_cb->mbhc_micbias_control)
  839. mbhc->mbhc_cb->mbhc_micbias_control(
  840. component, MIC_BIAS_2,
  841. MICB_DISABLE);
  842. if (mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic)
  843. mbhc->mbhc_cb->mbhc_micb_ctrl_thr_mic(
  844. component,
  845. MIC_BIAS_2, false);
  846. if (mbhc->mbhc_cb->set_micbias_value) {
  847. mbhc->mbhc_cb->set_micbias_value(component);
  848. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_MICB_CTRL, 0);
  849. }
  850. mbhc->micbias_enable = false;
  851. }
  852. if (mbhc->mbhc_cfg->detect_extn_cable &&
  853. ((plug_type == MBHC_PLUG_TYPE_HEADPHONE) ||
  854. (plug_type == MBHC_PLUG_TYPE_HEADSET)) &&
  855. !mbhc->hs_detect_work_stop) {
  856. WCD_MBHC_RSC_LOCK(mbhc);
  857. wcd_mbhc_hs_elec_irq(mbhc, WCD_MBHC_ELEC_HS_REM, true);
  858. WCD_MBHC_RSC_UNLOCK(mbhc);
  859. }
  860. /*
  861. * Enable ADC COMPLETE interrupt for HEADPHONE.
  862. * Btn release may happen after the correct work, ADC COMPLETE
  863. * interrupt needs to be captured to correct plug type.
  864. */
  865. if (plug_type == MBHC_PLUG_TYPE_HEADPHONE) {
  866. WCD_MBHC_RSC_LOCK(mbhc);
  867. wcd_mbhc_hs_elec_irq(mbhc, WCD_MBHC_ELEC_HS_INS,
  868. true);
  869. WCD_MBHC_RSC_UNLOCK(mbhc);
  870. }
  871. if (mbhc->mbhc_cb->hph_pull_down_ctrl)
  872. mbhc->mbhc_cb->hph_pull_down_ctrl(component, true);
  873. mbhc->mbhc_cb->lock_sleep(mbhc, false);
  874. pr_debug("%s: leave\n", __func__);
  875. }
  876. static irqreturn_t wcd_mbhc_adc_hs_rem_irq(int irq, void *data)
  877. {
  878. struct wcd_mbhc *mbhc = data;
  879. unsigned long timeout;
  880. int adc_threshold, output_mv, retry = 0;
  881. bool hphpa_on = false;
  882. u8 moisture_status = 0;
  883. pr_debug("%s: enter\n", __func__);
  884. WCD_MBHC_RSC_LOCK(mbhc);
  885. timeout = jiffies +
  886. msecs_to_jiffies(WCD_FAKE_REMOVAL_MIN_PERIOD_MS);
  887. adc_threshold = wcd_mbhc_adc_get_hs_thres(mbhc);
  888. /* Enable MICBIAS before checking for ADC Voltage */
  889. if (mbhc->mbhc_cb->mbhc_micbias_control)
  890. mbhc->mbhc_cb->mbhc_micbias_control(mbhc->component,
  891. MIC_BIAS_2, MICB_ENABLE);
  892. do {
  893. retry++;
  894. /*
  895. * read output_mv every 10ms to look for
  896. * any change in IN2_P
  897. */
  898. usleep_range(10000, 10100);
  899. output_mv = wcd_measure_adc_once(mbhc, MUX_CTL_IN2P);
  900. pr_debug("%s: Check for fake removal: output_mv %d\n",
  901. __func__, output_mv);
  902. if ((output_mv <= adc_threshold) &&
  903. retry > FAKE_REM_RETRY_ATTEMPTS) {
  904. pr_debug("%s: headset is NOT actually removed\n",
  905. __func__);
  906. if (mbhc->mbhc_cb->mbhc_micbias_control)
  907. mbhc->mbhc_cb->mbhc_micbias_control(
  908. mbhc->component, MIC_BIAS_2,
  909. MICB_DISABLE);
  910. goto exit;
  911. }
  912. } while (!time_after(jiffies, timeout));
  913. if (mbhc->mbhc_cb->mbhc_micbias_control)
  914. mbhc->mbhc_cb->mbhc_micbias_control(mbhc->component, MIC_BIAS_2,
  915. MICB_DISABLE);
  916. if (wcd_swch_level_remove(mbhc)) {
  917. pr_debug("%s: Switch level is low ", __func__);
  918. goto exit;
  919. }
  920. if (mbhc->mbhc_cfg->moisture_en ||
  921. mbhc->mbhc_cfg->moisture_duty_cycle_en) {
  922. if (mbhc->mbhc_cb->hph_pa_on_status)
  923. if (mbhc->mbhc_cb->hph_pa_on_status(mbhc->component)) {
  924. hphpa_on = true;
  925. WCD_MBHC_REG_UPDATE_BITS(
  926. WCD_MBHC_HPH_PA_EN, 0);
  927. }
  928. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_HPHR_GND, 1);
  929. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_HPHL_GND, 1);
  930. /* wait for 50ms to get moisture status */
  931. usleep_range(50000, 50100);
  932. WCD_MBHC_REG_READ(WCD_MBHC_MOISTURE_STATUS, moisture_status);
  933. }
  934. if ((mbhc->mbhc_cfg->moisture_en ||
  935. mbhc->mbhc_cfg->moisture_duty_cycle_en) && !moisture_status) {
  936. pr_debug("%s: moisture present in jack\n", __func__);
  937. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_L_DET_EN, 0);
  938. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_MECH_DETECTION_TYPE, 1);
  939. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_L_DET_EN, 1);
  940. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_FSM_EN, 0);
  941. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_BTN_ISRC_CTL, 0);
  942. mbhc->btn_press_intr = false;
  943. mbhc->is_btn_press = false;
  944. if (mbhc->current_plug == MBHC_PLUG_TYPE_HEADSET) {
  945. wcd_mbhc_report_plug(mbhc, 0, SND_JACK_HEADSET);
  946. extcon_set_state_sync(mbhc->extdev, EXTCON_JACK_MICROPHONE, 0);
  947. } else if (mbhc->current_plug == MBHC_PLUG_TYPE_HEADPHONE) {
  948. wcd_mbhc_report_plug(mbhc, 0, SND_JACK_HEADPHONE);
  949. extcon_set_state_sync(mbhc->extdev, EXTCON_JACK_HEADPHONE, 0);
  950. } else if (mbhc->current_plug == MBHC_PLUG_TYPE_GND_MIC_SWAP) {
  951. #if IS_ENABLED(CONFIG_AUDIO_QGKI)
  952. wcd_mbhc_report_plug(mbhc, 0, SND_JACK_UNSUPPORTED);
  953. #endif /* CONFIG_AUDIO_QGKI */
  954. extcon_set_state_sync(mbhc->extdev, EXTCON_MECHANICAL, 0);
  955. } else if (mbhc->current_plug == MBHC_PLUG_TYPE_HIGH_HPH) {
  956. wcd_mbhc_report_plug(mbhc, 0, SND_JACK_LINEOUT);
  957. extcon_set_state_sync(mbhc->extdev, EXTCON_JACK_LINE_OUT, 0);
  958. }
  959. } else {
  960. /*
  961. * ADC COMPLETE and ELEC_REM interrupts are both enabled for
  962. * HEADPHONE, need to reject the ADC COMPLETE interrupt which
  963. * follows ELEC_REM one when HEADPHONE is removed.
  964. */
  965. if (mbhc->current_plug == MBHC_PLUG_TYPE_HEADPHONE)
  966. mbhc->extn_cable_hph_rem = true;
  967. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 0);
  968. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_MODE, 0);
  969. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ADC_EN, 0);
  970. wcd_mbhc_elec_hs_report_unplug(mbhc);
  971. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_BTN_ISRC_CTL, 0);
  972. if (hphpa_on) {
  973. hphpa_on = false;
  974. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_HPH_PA_EN, 3);
  975. }
  976. }
  977. exit:
  978. WCD_MBHC_RSC_UNLOCK(mbhc);
  979. pr_debug("%s: leave\n", __func__);
  980. return IRQ_HANDLED;
  981. }
  982. static irqreturn_t wcd_mbhc_adc_hs_ins_irq(int irq, void *data)
  983. {
  984. struct wcd_mbhc *mbhc = data;
  985. u8 clamp_state = 0;
  986. u8 clamp_retry = WCD_MBHC_FAKE_INS_RETRY;
  987. pr_debug("%s: enter\n", __func__);
  988. /*
  989. * ADC COMPLETE and ELEC_REM interrupts are both enabled for HEADPHONE,
  990. * need to reject the ADC COMPLETE interrupt which follows ELEC_REM one
  991. * when HEADPHONE is removed.
  992. */
  993. if (mbhc->extn_cable_hph_rem == true) {
  994. mbhc->extn_cable_hph_rem = false;
  995. pr_debug("%s: leave\n", __func__);
  996. return IRQ_HANDLED;
  997. }
  998. do {
  999. WCD_MBHC_REG_READ(WCD_MBHC_IN2P_CLAMP_STATE, clamp_state);
  1000. if (clamp_state) {
  1001. pr_debug("%s: fake insertion irq, leave\n", __func__);
  1002. return IRQ_HANDLED;
  1003. }
  1004. /*
  1005. * check clamp for 120ms but at 30ms chunks to leave
  1006. * room for other interrupts to be processed
  1007. */
  1008. usleep_range(30000, 30100);
  1009. } while (--clamp_retry);
  1010. WCD_MBHC_RSC_LOCK(mbhc);
  1011. /*
  1012. * If current plug is headphone then there is no chance to
  1013. * get ADC complete interrupt, so connected cable should be
  1014. * headset not headphone.
  1015. */
  1016. if (mbhc->current_plug == MBHC_PLUG_TYPE_HEADPHONE) {
  1017. wcd_mbhc_hs_elec_irq(mbhc, WCD_MBHC_ELEC_HS_INS, false);
  1018. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_DETECTION_DONE, 1);
  1019. wcd_mbhc_find_plug_and_report(mbhc, MBHC_PLUG_TYPE_HEADSET);
  1020. WCD_MBHC_RSC_UNLOCK(mbhc);
  1021. return IRQ_HANDLED;
  1022. }
  1023. if (!mbhc->mbhc_cfg->detect_extn_cable) {
  1024. pr_debug("%s: Returning as Extension cable feature not enabled\n",
  1025. __func__);
  1026. WCD_MBHC_RSC_UNLOCK(mbhc);
  1027. return IRQ_HANDLED;
  1028. }
  1029. pr_debug("%s: Disable electrical headset insertion interrupt\n",
  1030. __func__);
  1031. wcd_mbhc_hs_elec_irq(mbhc, WCD_MBHC_ELEC_HS_INS, false);
  1032. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_SCHMT_ISRC, 0);
  1033. WCD_MBHC_REG_UPDATE_BITS(WCD_MBHC_ELECT_ISRC_EN, 0);
  1034. mbhc->is_extn_cable = true;
  1035. mbhc->btn_press_intr = false;
  1036. mbhc->force_linein = false;
  1037. wcd_mbhc_adc_detect_plug_type(mbhc);
  1038. WCD_MBHC_RSC_UNLOCK(mbhc);
  1039. pr_debug("%s: leave\n", __func__);
  1040. return IRQ_HANDLED;
  1041. }
  1042. static struct wcd_mbhc_fn mbhc_fn = {
  1043. .wcd_mbhc_hs_ins_irq = wcd_mbhc_adc_hs_ins_irq,
  1044. .wcd_mbhc_hs_rem_irq = wcd_mbhc_adc_hs_rem_irq,
  1045. .wcd_mbhc_detect_plug_type = wcd_mbhc_adc_detect_plug_type,
  1046. .wcd_mbhc_detect_anc_plug_type = wcd_mbhc_adc_detect_anc_plug_type,
  1047. .wcd_cancel_hs_detect_plug = wcd_cancel_hs_detect_plug,
  1048. };
  1049. /* Function: wcd_mbhc_adc_init
  1050. * @mbhc: MBHC function pointer
  1051. * Description: Initialize MBHC ADC related function pointers to MBHC structure
  1052. */
  1053. void wcd_mbhc_adc_init(struct wcd_mbhc *mbhc)
  1054. {
  1055. if (!mbhc) {
  1056. pr_err("%s: mbhc is NULL\n", __func__);
  1057. return;
  1058. }
  1059. mbhc->mbhc_fn = &mbhc_fn;
  1060. INIT_WORK(&mbhc->correct_plug_swch, wcd_correct_swch_plug);
  1061. }
  1062. EXPORT_SYMBOL(wcd_mbhc_adc_init);