wcd-mbhc-adc.c 32 KB

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