audio_cal_utils.c 27 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2014-2020, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/slab.h>
  6. #include <linux/fs.h>
  7. #include <linux/module.h>
  8. #include <linux/miscdevice.h>
  9. #include <linux/uaccess.h>
  10. #include <linux/mutex.h>
  11. #include <dsp/audio_cal_utils.h>
  12. spinlock_t cal_lock;
  13. static int unmap_memory(struct cal_type_data *cal_type,
  14. struct cal_block_data *cal_block);
  15. size_t get_cal_info_size(int32_t cal_type)
  16. {
  17. size_t size = 0;
  18. size_t size1 = 0, size2 = 0;
  19. switch (cal_type) {
  20. case CVP_VOC_RX_TOPOLOGY_CAL_TYPE:
  21. size = sizeof(struct audio_cal_info_voc_top);
  22. break;
  23. case CVP_VOC_TX_TOPOLOGY_CAL_TYPE:
  24. size = sizeof(struct audio_cal_info_voc_top);
  25. break;
  26. case CVP_VOCPROC_STATIC_CAL_TYPE:
  27. size = sizeof(struct audio_cal_info_vocproc);
  28. break;
  29. case CVP_VOCPROC_DYNAMIC_CAL_TYPE:
  30. size = sizeof(struct audio_cal_info_vocvol);
  31. break;
  32. case CVS_VOCSTRM_STATIC_CAL_TYPE:
  33. size = 0;
  34. break;
  35. case CVP_VOCDEV_CFG_CAL_TYPE:
  36. size = sizeof(struct audio_cal_info_vocdev_cfg);
  37. break;
  38. case CVP_VOCPROC_STATIC_COL_CAL_TYPE:
  39. size = sizeof(struct audio_cal_info_voc_col);
  40. break;
  41. case CVP_VOCPROC_DYNAMIC_COL_CAL_TYPE:
  42. size = sizeof(struct audio_cal_info_voc_col);
  43. break;
  44. case CVS_VOCSTRM_STATIC_COL_CAL_TYPE:
  45. size = sizeof(struct audio_cal_info_voc_col);
  46. break;
  47. case ADM_TOPOLOGY_CAL_TYPE:
  48. case ADM_LSM_TOPOLOGY_CAL_TYPE:
  49. size = sizeof(struct audio_cal_info_adm_top);
  50. break;
  51. case ADM_CUST_TOPOLOGY_CAL_TYPE:
  52. case CORE_CUSTOM_TOPOLOGIES_CAL_TYPE:
  53. size = 0;
  54. break;
  55. case ADM_AUDPROC_CAL_TYPE:
  56. case ADM_LSM_AUDPROC_CAL_TYPE:
  57. case ADM_LSM_AUDPROC_PERSISTENT_CAL_TYPE:
  58. case ADM_AUDPROC_PERSISTENT_CAL_TYPE:
  59. size = sizeof(struct audio_cal_info_audproc);
  60. break;
  61. case ADM_AUDVOL_CAL_TYPE:
  62. case ADM_RTAC_AUDVOL_CAL_TYPE:
  63. size = sizeof(struct audio_cal_info_audvol);
  64. break;
  65. case ASM_TOPOLOGY_CAL_TYPE:
  66. size = sizeof(struct audio_cal_info_asm_top);
  67. break;
  68. case ASM_CUST_TOPOLOGY_CAL_TYPE:
  69. size = 0;
  70. break;
  71. case ASM_AUDSTRM_CAL_TYPE:
  72. size = sizeof(struct audio_cal_info_audstrm);
  73. break;
  74. case AFE_TOPOLOGY_CAL_TYPE:
  75. case AFE_LSM_TOPOLOGY_CAL_TYPE:
  76. size = sizeof(struct audio_cal_info_afe_top);
  77. break;
  78. case AFE_CUST_TOPOLOGY_CAL_TYPE:
  79. size = 0;
  80. break;
  81. case AFE_COMMON_RX_CAL_TYPE:
  82. size = sizeof(struct audio_cal_info_afe);
  83. break;
  84. case AFE_COMMON_TX_CAL_TYPE:
  85. case AFE_LSM_TX_CAL_TYPE:
  86. size = sizeof(struct audio_cal_info_afe);
  87. break;
  88. case AFE_FB_SPKR_PROT_CAL_TYPE:
  89. size = sizeof(struct audio_cal_info_spk_prot_cfg);
  90. break;
  91. case AFE_FB_SPKR_PROT_TH_VI_CAL_TYPE:
  92. /*
  93. * Since get and set parameter structures are different in size
  94. * use the maximum size of get and set parameter structure
  95. */
  96. size1 = max(sizeof(struct audio_cal_info_sp_th_vi_ftm_cfg),
  97. sizeof(struct audio_cal_info_sp_th_vi_param));
  98. size2 = max(sizeof(struct audio_cal_info_sp_th_vi_v_vali_cfg),
  99. sizeof(struct audio_cal_info_sp_th_vi_v_vali_param));
  100. size = max(size1, size2);
  101. break;
  102. case AFE_FB_SPKR_PROT_EX_VI_CAL_TYPE:
  103. /*
  104. * Since get and set parameter structures are different in size
  105. * use the maximum size of get and set parameter structure
  106. */
  107. size = max(sizeof(struct audio_cal_info_sp_ex_vi_ftm_cfg),
  108. sizeof(struct audio_cal_info_sp_ex_vi_param));
  109. break;
  110. case AFE_FB_SPKR_PROT_V4_EX_VI_CAL_TYPE:
  111. size = sizeof(struct audio_cal_info_sp_v4_ex_vi_param);
  112. break;
  113. case AFE_ANC_CAL_TYPE:
  114. size = 0;
  115. break;
  116. case AFE_AANC_CAL_TYPE:
  117. size = sizeof(struct audio_cal_info_aanc);
  118. break;
  119. case AFE_HW_DELAY_CAL_TYPE:
  120. size = sizeof(struct audio_cal_info_hw_delay);
  121. break;
  122. case AFE_SIDETONE_CAL_TYPE:
  123. size = sizeof(struct audio_cal_info_sidetone);
  124. break;
  125. case AFE_SIDETONE_IIR_CAL_TYPE:
  126. size = sizeof(struct audio_cal_info_sidetone_iir);
  127. break;
  128. case LSM_CUST_TOPOLOGY_CAL_TYPE:
  129. size = 0;
  130. break;
  131. case LSM_TOPOLOGY_CAL_TYPE:
  132. size = sizeof(struct audio_cal_info_lsm_top);
  133. break;
  134. case ULP_LSM_TOPOLOGY_ID_CAL_TYPE:
  135. size = sizeof(struct audio_cal_info_lsm_top);
  136. break;
  137. case LSM_CAL_TYPE:
  138. size = sizeof(struct audio_cal_info_lsm);
  139. break;
  140. case ADM_RTAC_INFO_CAL_TYPE:
  141. size = 0;
  142. break;
  143. case VOICE_RTAC_INFO_CAL_TYPE:
  144. size = 0;
  145. break;
  146. case ADM_RTAC_APR_CAL_TYPE:
  147. size = 0;
  148. break;
  149. case ASM_RTAC_APR_CAL_TYPE:
  150. size = 0;
  151. break;
  152. case VOICE_RTAC_APR_CAL_TYPE:
  153. size = 0;
  154. break;
  155. case MAD_CAL_TYPE:
  156. size = 0;
  157. break;
  158. case ULP_AFE_CAL_TYPE:
  159. size = sizeof(struct audio_cal_info_afe);
  160. break;
  161. case ULP_LSM_CAL_TYPE:
  162. size = sizeof(struct audio_cal_info_lsm);
  163. break;
  164. case AUDIO_CORE_METAINFO_CAL_TYPE:
  165. size = sizeof(struct audio_cal_info_metainfo);
  166. break;
  167. case SRS_TRUMEDIA_CAL_TYPE:
  168. size = 0;
  169. break;
  170. default:
  171. pr_err("%s:Invalid cal type %d!",
  172. __func__, cal_type);
  173. }
  174. return size;
  175. }
  176. size_t get_user_cal_type_size(int32_t cal_type)
  177. {
  178. size_t size = 0;
  179. switch (cal_type) {
  180. case CVP_VOC_RX_TOPOLOGY_CAL_TYPE:
  181. size = sizeof(struct audio_cal_type_voc_top);
  182. break;
  183. case CVP_VOC_TX_TOPOLOGY_CAL_TYPE:
  184. size = sizeof(struct audio_cal_type_voc_top);
  185. break;
  186. case CVP_VOCPROC_STATIC_CAL_TYPE:
  187. size = sizeof(struct audio_cal_type_vocproc);
  188. break;
  189. case CVP_VOCPROC_DYNAMIC_CAL_TYPE:
  190. size = sizeof(struct audio_cal_type_vocvol);
  191. break;
  192. case CVS_VOCSTRM_STATIC_CAL_TYPE:
  193. size = sizeof(struct audio_cal_type_basic);
  194. break;
  195. case CVP_VOCDEV_CFG_CAL_TYPE:
  196. size = sizeof(struct audio_cal_type_vocdev_cfg);
  197. break;
  198. case CVP_VOCPROC_STATIC_COL_CAL_TYPE:
  199. case CVP_VOCPROC_DYNAMIC_COL_CAL_TYPE:
  200. case CVS_VOCSTRM_STATIC_COL_CAL_TYPE:
  201. size = sizeof(struct audio_cal_type_voc_col);
  202. break;
  203. case ADM_TOPOLOGY_CAL_TYPE:
  204. case ADM_LSM_TOPOLOGY_CAL_TYPE:
  205. size = sizeof(struct audio_cal_type_adm_top);
  206. break;
  207. case ADM_CUST_TOPOLOGY_CAL_TYPE:
  208. case CORE_CUSTOM_TOPOLOGIES_CAL_TYPE:
  209. size = sizeof(struct audio_cal_type_basic);
  210. break;
  211. case ADM_AUDPROC_CAL_TYPE:
  212. case ADM_LSM_AUDPROC_CAL_TYPE:
  213. case ADM_LSM_AUDPROC_PERSISTENT_CAL_TYPE:
  214. case ADM_AUDPROC_PERSISTENT_CAL_TYPE:
  215. size = sizeof(struct audio_cal_type_audproc);
  216. break;
  217. case ADM_AUDVOL_CAL_TYPE:
  218. case ADM_RTAC_AUDVOL_CAL_TYPE:
  219. size = sizeof(struct audio_cal_type_audvol);
  220. break;
  221. case ASM_TOPOLOGY_CAL_TYPE:
  222. size = sizeof(struct audio_cal_type_asm_top);
  223. break;
  224. case ASM_CUST_TOPOLOGY_CAL_TYPE:
  225. size = sizeof(struct audio_cal_type_basic);
  226. break;
  227. case ASM_AUDSTRM_CAL_TYPE:
  228. size = sizeof(struct audio_cal_type_audstrm);
  229. break;
  230. case AFE_TOPOLOGY_CAL_TYPE:
  231. case AFE_LSM_TOPOLOGY_CAL_TYPE:
  232. size = sizeof(struct audio_cal_type_afe_top);
  233. break;
  234. case AFE_CUST_TOPOLOGY_CAL_TYPE:
  235. size = sizeof(struct audio_cal_type_basic);
  236. break;
  237. case AFE_COMMON_RX_CAL_TYPE:
  238. size = sizeof(struct audio_cal_type_afe);
  239. break;
  240. case AFE_COMMON_TX_CAL_TYPE:
  241. case AFE_LSM_TX_CAL_TYPE:
  242. size = sizeof(struct audio_cal_type_afe);
  243. break;
  244. case AFE_FB_SPKR_PROT_CAL_TYPE:
  245. size = sizeof(struct audio_cal_type_fb_spk_prot_cfg);
  246. break;
  247. case AFE_FB_SPKR_PROT_TH_VI_CAL_TYPE:
  248. /*
  249. * Since get and set parameter structures are different in size
  250. * use the maximum size of get and set parameter structure
  251. */
  252. size = max(sizeof(struct audio_cal_type_sp_th_vi_ftm_cfg),
  253. sizeof(struct audio_cal_type_sp_th_vi_param));
  254. break;
  255. case AFE_FB_SPKR_PROT_EX_VI_CAL_TYPE:
  256. /*
  257. * Since get and set parameter structures are different in size
  258. * use the maximum size of get and set parameter structure
  259. */
  260. size = max(sizeof(struct audio_cal_type_sp_ex_vi_ftm_cfg),
  261. sizeof(struct audio_cal_type_sp_ex_vi_param));
  262. break;
  263. case AFE_FB_SPKR_PROT_V4_EX_VI_CAL_TYPE:
  264. size = sizeof(struct audio_cal_type_sp_v4_ex_vi_param);
  265. break;
  266. case AFE_ANC_CAL_TYPE:
  267. size = 0;
  268. break;
  269. case AFE_AANC_CAL_TYPE:
  270. size = sizeof(struct audio_cal_type_aanc);
  271. break;
  272. case AFE_HW_DELAY_CAL_TYPE:
  273. size = sizeof(struct audio_cal_type_hw_delay);
  274. break;
  275. case AFE_SIDETONE_CAL_TYPE:
  276. size = sizeof(struct audio_cal_type_sidetone);
  277. break;
  278. case AFE_SIDETONE_IIR_CAL_TYPE:
  279. size = sizeof(struct audio_cal_type_sidetone_iir);
  280. break;
  281. case LSM_CUST_TOPOLOGY_CAL_TYPE:
  282. size = sizeof(struct audio_cal_type_basic);
  283. break;
  284. case LSM_TOPOLOGY_CAL_TYPE:
  285. size = sizeof(struct audio_cal_type_lsm_top);
  286. break;
  287. case ULP_LSM_TOPOLOGY_ID_CAL_TYPE:
  288. size = sizeof(struct audio_cal_type_lsm_top);
  289. break;
  290. case LSM_CAL_TYPE:
  291. size = sizeof(struct audio_cal_type_lsm);
  292. break;
  293. case ADM_RTAC_INFO_CAL_TYPE:
  294. size = 0;
  295. break;
  296. case VOICE_RTAC_INFO_CAL_TYPE:
  297. size = 0;
  298. break;
  299. case ADM_RTAC_APR_CAL_TYPE:
  300. size = 0;
  301. break;
  302. case ASM_RTAC_APR_CAL_TYPE:
  303. size = 0;
  304. break;
  305. case VOICE_RTAC_APR_CAL_TYPE:
  306. size = 0;
  307. break;
  308. case MAD_CAL_TYPE:
  309. size = 0;
  310. break;
  311. case ULP_AFE_CAL_TYPE:
  312. size = sizeof(struct audio_cal_type_afe);
  313. break;
  314. case ULP_LSM_CAL_TYPE:
  315. size = sizeof(struct audio_cal_type_lsm);
  316. break;
  317. case AUDIO_CORE_METAINFO_CAL_TYPE:
  318. size = sizeof(struct audio_cal_type_metainfo);
  319. break;
  320. case SRS_TRUMEDIA_CAL_TYPE:
  321. size = 0;
  322. break;
  323. default:
  324. pr_err("%s:Invalid cal type %d!",
  325. __func__, cal_type);
  326. }
  327. return size;
  328. }
  329. int32_t cal_utils_get_cal_type_version(void *cal_type_data)
  330. {
  331. struct audio_cal_type_basic *data = NULL;
  332. data = (struct audio_cal_type_basic *)cal_type_data;
  333. return data->cal_hdr.version;
  334. }
  335. static struct cal_type_data *create_cal_type_data(
  336. struct cal_type_info *info)
  337. {
  338. struct cal_type_data *cal_type = NULL;
  339. if ((info->reg.cal_type < 0) ||
  340. (info->reg.cal_type >= MAX_CAL_TYPES)) {
  341. pr_err("%s: cal type %d is Invalid!\n",
  342. __func__, info->reg.cal_type);
  343. goto done;
  344. }
  345. if (info->cal_util_callbacks.match_block == NULL) {
  346. pr_err("%s: cal type %d no method to match blocks!\n",
  347. __func__, info->reg.cal_type);
  348. goto done;
  349. }
  350. cal_type = kmalloc(sizeof(*cal_type), GFP_KERNEL);
  351. if (cal_type == NULL)
  352. goto done;
  353. INIT_LIST_HEAD(&cal_type->cal_blocks);
  354. mutex_init(&cal_type->lock);
  355. memcpy(&cal_type->info, info,
  356. sizeof(cal_type->info));
  357. done:
  358. return cal_type;
  359. }
  360. /**
  361. * cal_utils_create_cal_types
  362. *
  363. * @num_cal_types: number of types
  364. * @cal_type: pointer to the cal types pointer
  365. * @info: pointer to info
  366. *
  367. * Returns 0 on success, EINVAL otherwise
  368. */
  369. int cal_utils_create_cal_types(int num_cal_types,
  370. struct cal_type_data **cal_type,
  371. struct cal_type_info *info)
  372. {
  373. int ret = 0;
  374. int i;
  375. pr_debug("%s\n", __func__);
  376. if (cal_type == NULL) {
  377. pr_err("%s: cal_type is NULL!\n", __func__);
  378. ret = -EINVAL;
  379. goto done;
  380. } else if (info == NULL) {
  381. pr_err("%s: info is NULL!\n", __func__);
  382. ret = -EINVAL;
  383. goto done;
  384. } else if ((num_cal_types <= 0) ||
  385. (num_cal_types > MAX_CAL_TYPES)) {
  386. pr_err("%s: num_cal_types of %d is Invalid!\n",
  387. __func__, num_cal_types);
  388. ret = -EINVAL;
  389. goto done;
  390. }
  391. for (i = 0; i < num_cal_types; i++) {
  392. if ((info[i].reg.cal_type < 0) ||
  393. (info[i].reg.cal_type >= MAX_CAL_TYPES)) {
  394. pr_err("%s: cal type %d at index %d is Invalid!\n",
  395. __func__, info[i].reg.cal_type, i);
  396. ret = -EINVAL;
  397. goto done;
  398. }
  399. cal_type[i] = create_cal_type_data(&info[i]);
  400. if (cal_type[i] == NULL) {
  401. pr_err("%s: Could not allocate cal_type of index %d!\n",
  402. __func__, i);
  403. ret = -EINVAL;
  404. goto done;
  405. }
  406. ret = audio_cal_register(1, &info[i].reg);
  407. if (ret < 0) {
  408. pr_err("%s: audio_cal_register failed, ret = %d!\n",
  409. __func__, ret);
  410. ret = -EINVAL;
  411. goto done;
  412. }
  413. pr_debug("%s: cal type %d at index %d!\n",
  414. __func__, info[i].reg.cal_type, i);
  415. }
  416. done:
  417. return ret;
  418. }
  419. EXPORT_SYMBOL(cal_utils_create_cal_types);
  420. static void delete_cal_block(struct cal_block_data *cal_block)
  421. {
  422. pr_debug("%s\n", __func__);
  423. if (cal_block == NULL)
  424. goto done;
  425. list_del(&cal_block->list);
  426. kfree(cal_block->client_info);
  427. cal_block->client_info = NULL;
  428. kfree(cal_block->cal_info);
  429. cal_block->cal_info = NULL;
  430. if (cal_block->map_data.dma_buf != NULL) {
  431. if (cal_block->cma_mem)
  432. msm_audio_ion_free_cma(cal_block->map_data.dma_buf);
  433. else
  434. msm_audio_ion_free(cal_block->map_data.dma_buf);
  435. cal_block->map_data.dma_buf = NULL;
  436. }
  437. kfree(cal_block);
  438. done:
  439. return;
  440. }
  441. static void destroy_all_cal_blocks(struct cal_type_data *cal_type)
  442. {
  443. int ret = 0;
  444. struct list_head *ptr, *next;
  445. struct cal_block_data *cal_block;
  446. list_for_each_safe(ptr, next,
  447. &cal_type->cal_blocks) {
  448. cal_block = list_entry(ptr,
  449. struct cal_block_data, list);
  450. ret = unmap_memory(cal_type, cal_block);
  451. if (ret < 0) {
  452. pr_err("%s: unmap_memory failed, cal type %d, ret = %d!\n",
  453. __func__,
  454. cal_type->info.reg.cal_type,
  455. ret);
  456. }
  457. delete_cal_block(cal_block);
  458. cal_block = NULL;
  459. }
  460. }
  461. static void destroy_cal_type_data(struct cal_type_data *cal_type)
  462. {
  463. if (cal_type == NULL)
  464. goto done;
  465. destroy_all_cal_blocks(cal_type);
  466. list_del(&cal_type->cal_blocks);
  467. kfree(cal_type);
  468. done:
  469. return;
  470. }
  471. /**
  472. * cal_utils_destroy_cal_types -
  473. * Destroys cal types and deregister from cal info
  474. *
  475. * @num_cal_types: number of cal types
  476. * @cal_type: cal type pointer with cal info
  477. *
  478. */
  479. void cal_utils_destroy_cal_types(int num_cal_types,
  480. struct cal_type_data **cal_type)
  481. {
  482. int i;
  483. pr_debug("%s\n", __func__);
  484. if (cal_type == NULL) {
  485. pr_err("%s: cal_type is NULL!\n", __func__);
  486. goto done;
  487. } else if ((num_cal_types <= 0) ||
  488. (num_cal_types > MAX_CAL_TYPES)) {
  489. pr_err("%s: num_cal_types of %d is Invalid!\n",
  490. __func__, num_cal_types);
  491. goto done;
  492. }
  493. for (i = 0; i < num_cal_types; i++) {
  494. audio_cal_deregister(1, &cal_type[i]->info.reg);
  495. destroy_cal_type_data(cal_type[i]);
  496. cal_type[i] = NULL;
  497. }
  498. done:
  499. return;
  500. }
  501. EXPORT_SYMBOL(cal_utils_destroy_cal_types);
  502. /**
  503. * cal_utils_get_only_cal_block
  504. *
  505. * @cal_type: pointer to the cal type
  506. *
  507. * Returns cal_block structure
  508. */
  509. struct cal_block_data *cal_utils_get_only_cal_block(
  510. struct cal_type_data *cal_type)
  511. {
  512. struct list_head *ptr, *next;
  513. struct cal_block_data *cal_block = NULL;
  514. if (cal_type == NULL)
  515. goto done;
  516. list_for_each_safe(ptr, next,
  517. &cal_type->cal_blocks) {
  518. cal_block = list_entry(ptr,
  519. struct cal_block_data, list);
  520. break;
  521. }
  522. done:
  523. return cal_block;
  524. }
  525. EXPORT_SYMBOL(cal_utils_get_only_cal_block);
  526. /**
  527. * cal_utils_get_only_cal_block
  528. *
  529. * @cal_block: pointer to cal block struct
  530. * @user_data: pointer to user data
  531. *
  532. * Returns true on match
  533. */
  534. bool cal_utils_match_buf_num(struct cal_block_data *cal_block,
  535. void *user_data)
  536. {
  537. bool ret = false;
  538. struct audio_cal_type_basic *data = user_data;
  539. if (cal_block->buffer_number == data->cal_hdr.buffer_number)
  540. ret = true;
  541. return ret;
  542. }
  543. EXPORT_SYMBOL(cal_utils_match_buf_num);
  544. static struct cal_block_data *get_matching_cal_block(
  545. struct cal_type_data *cal_type,
  546. void *data)
  547. {
  548. struct list_head *ptr, *next;
  549. struct cal_block_data *cal_block = NULL;
  550. list_for_each_safe(ptr, next,
  551. &cal_type->cal_blocks) {
  552. cal_block = list_entry(ptr,
  553. struct cal_block_data, list);
  554. if (cal_type->info.cal_util_callbacks.
  555. match_block(cal_block, data))
  556. return cal_block;
  557. }
  558. return NULL;
  559. }
  560. static int cal_block_ion_alloc(struct cal_block_data *cal_block)
  561. {
  562. int ret = 0;
  563. if (cal_block == NULL) {
  564. pr_err("%s: cal_block is NULL!\n", __func__);
  565. ret = -EINVAL;
  566. goto done;
  567. }
  568. if (cal_block->cma_mem) {
  569. ret = msm_audio_ion_import_cma(&cal_block->map_data.dma_buf,
  570. cal_block->map_data.ion_map_handle,
  571. NULL, 0,
  572. &cal_block->cal_data.paddr,
  573. &cal_block->map_data.map_size,
  574. &cal_block->cal_data.kvaddr);
  575. } else {
  576. ret = msm_audio_ion_import(&cal_block->map_data.dma_buf,
  577. cal_block->map_data.ion_map_handle,
  578. NULL, 0,
  579. &cal_block->cal_data.paddr,
  580. &cal_block->map_data.map_size,
  581. &cal_block->cal_data.kvaddr);
  582. }
  583. if (ret) {
  584. pr_err("%s: audio ION import failed, rc = %d\n",
  585. __func__, ret);
  586. ret = -ENOMEM;
  587. goto done;
  588. }
  589. done:
  590. return ret;
  591. }
  592. static struct cal_block_data *create_cal_block(struct cal_type_data *cal_type,
  593. struct audio_cal_type_basic *basic_cal,
  594. size_t client_info_size, void *client_info)
  595. {
  596. struct cal_block_data *cal_block = NULL;
  597. if (cal_type == NULL) {
  598. pr_err("%s: cal_type is NULL!\n", __func__);
  599. goto done;
  600. } else if (basic_cal == NULL) {
  601. pr_err("%s: basic_cal is NULL!\n", __func__);
  602. goto done;
  603. }
  604. cal_block = kzalloc(sizeof(*cal_block),
  605. GFP_KERNEL);
  606. if (cal_block == NULL)
  607. goto done;
  608. INIT_LIST_HEAD(&cal_block->list);
  609. cal_block->map_data.ion_map_handle = basic_cal->cal_data.mem_handle;
  610. cal_block->cma_mem = basic_cal->cal_data.cma_mem;
  611. if (basic_cal->cal_data.mem_handle > 0) {
  612. if (cal_block_ion_alloc(cal_block)) {
  613. pr_err("%s: cal_block_ion_alloc failed!\n",
  614. __func__);
  615. goto err;
  616. }
  617. }
  618. if (client_info_size > 0) {
  619. cal_block->client_info_size = client_info_size;
  620. cal_block->client_info = kmalloc(client_info_size, GFP_KERNEL);
  621. if (cal_block->client_info == NULL) {
  622. pr_err("%s: could not allocats client_info!\n",
  623. __func__);
  624. goto err;
  625. }
  626. if (client_info != NULL)
  627. memcpy(cal_block->client_info, client_info,
  628. client_info_size);
  629. }
  630. cal_block->cal_info = kzalloc(
  631. get_cal_info_size(cal_type->info.reg.cal_type),
  632. GFP_KERNEL);
  633. if (cal_block->cal_info == NULL) {
  634. pr_err("%s: could not allocats cal_info!\n",
  635. __func__);
  636. goto err;
  637. }
  638. cal_block->buffer_number = basic_cal->cal_hdr.buffer_number;
  639. list_add_tail(&cal_block->list, &cal_type->cal_blocks);
  640. pr_debug("%s: created block for cal type %d, buf num %d, map handle %d, map size %zd paddr 0x%pK!\n",
  641. __func__, cal_type->info.reg.cal_type,
  642. cal_block->buffer_number,
  643. cal_block->map_data.ion_map_handle,
  644. cal_block->map_data.map_size,
  645. &cal_block->cal_data.paddr);
  646. done:
  647. return cal_block;
  648. err:
  649. kfree(cal_block->cal_info);
  650. cal_block->cal_info = NULL;
  651. kfree(cal_block->client_info);
  652. cal_block->client_info = NULL;
  653. kfree(cal_block);
  654. cal_block = NULL;
  655. return cal_block;
  656. }
  657. void cal_utils_clear_cal_block_q6maps(int num_cal_types,
  658. struct cal_type_data **cal_type)
  659. {
  660. int i = 0;
  661. struct list_head *ptr, *next;
  662. struct cal_block_data *cal_block;
  663. pr_debug("%s\n", __func__);
  664. if (cal_type == NULL) {
  665. pr_err("%s: cal_type is NULL!\n", __func__);
  666. goto done;
  667. } else if ((num_cal_types <= 0) ||
  668. (num_cal_types > MAX_CAL_TYPES)) {
  669. pr_err("%s: num_cal_types of %d is Invalid!\n",
  670. __func__, num_cal_types);
  671. goto done;
  672. }
  673. for (; i < num_cal_types; i++) {
  674. if (cal_type[i] == NULL)
  675. continue;
  676. mutex_lock(&cal_type[i]->lock);
  677. list_for_each_safe(ptr, next,
  678. &cal_type[i]->cal_blocks) {
  679. cal_block = list_entry(ptr,
  680. struct cal_block_data, list);
  681. cal_block->map_data.q6map_handle = 0;
  682. }
  683. mutex_unlock(&cal_type[i]->lock);
  684. }
  685. done:
  686. return;
  687. }
  688. static int realloc_memory(struct cal_block_data *cal_block)
  689. {
  690. int ret = 0;
  691. if (cal_block->cma_mem)
  692. msm_audio_ion_free_cma(cal_block->map_data.dma_buf);
  693. else
  694. msm_audio_ion_free(cal_block->map_data.dma_buf);
  695. cal_block->map_data.dma_buf = NULL;
  696. cal_block->cal_data.size = 0;
  697. ret = cal_block_ion_alloc(cal_block);
  698. if (ret < 0)
  699. pr_err("%s: realloc_memory failed!\n",
  700. __func__);
  701. return ret;
  702. }
  703. static int map_memory(struct cal_type_data *cal_type,
  704. struct cal_block_data *cal_block)
  705. {
  706. int ret = 0;
  707. if (cal_type->info.cal_util_callbacks.map_cal != NULL) {
  708. if ((cal_block->map_data.ion_map_handle < 0) ||
  709. (cal_block->map_data.map_size <= 0) ||
  710. (cal_block->map_data.q6map_handle != 0)) {
  711. goto done;
  712. }
  713. pr_debug("%s: cal type %d call map\n",
  714. __func__, cal_type->info.reg.cal_type);
  715. ret = cal_type->info.cal_util_callbacks.
  716. map_cal(cal_type->info.reg.cal_type, cal_block);
  717. if (ret < 0) {
  718. pr_err("%s: map_cal failed, cal type %d, ret = %d!\n",
  719. __func__, cal_type->info.reg.cal_type,
  720. ret);
  721. goto done;
  722. }
  723. }
  724. done:
  725. return ret;
  726. }
  727. static int unmap_memory(struct cal_type_data *cal_type,
  728. struct cal_block_data *cal_block)
  729. {
  730. int ret = 0;
  731. if (cal_type->info.cal_util_callbacks.unmap_cal != NULL) {
  732. if ((cal_block->map_data.ion_map_handle < 0) ||
  733. (cal_block->map_data.map_size <= 0) ||
  734. (cal_block->map_data.q6map_handle == 0)) {
  735. goto done;
  736. }
  737. pr_debug("%s: cal type %d call unmap\n",
  738. __func__, cal_type->info.reg.cal_type);
  739. ret = cal_type->info.cal_util_callbacks.
  740. unmap_cal(cal_type->info.reg.cal_type, cal_block);
  741. if (ret < 0) {
  742. pr_err("%s: unmap_cal failed, cal type %d, ret = %d!\n",
  743. __func__, cal_type->info.reg.cal_type,
  744. ret);
  745. goto done;
  746. }
  747. }
  748. done:
  749. return ret;
  750. }
  751. /**
  752. * cal_utils_alloc_cal
  753. *
  754. * @data_size: size of data to allocate
  755. * @data: data pointer
  756. * @cal_type: pointer to the cal type
  757. * @client_info_size: client info size
  758. * @client_info: pointer to client info
  759. *
  760. * Returns 0 on success, appropriate error code otherwise
  761. */
  762. int cal_utils_alloc_cal(size_t data_size, void *data,
  763. struct cal_type_data *cal_type,
  764. size_t client_info_size, void *client_info)
  765. {
  766. int ret = 0;
  767. struct cal_block_data *cal_block;
  768. struct audio_cal_type_alloc *alloc_data = data;
  769. pr_debug("%s\n", __func__);
  770. if (cal_type == NULL) {
  771. pr_err("%s: cal_type is NULL!\n",
  772. __func__);
  773. ret = -EINVAL;
  774. goto done;
  775. }
  776. if (data_size < sizeof(struct audio_cal_type_alloc)) {
  777. pr_err("%s: data_size of %zd does not equal alloc struct size of %zd!\n",
  778. __func__, data_size,
  779. sizeof(struct audio_cal_type_alloc));
  780. ret = -EINVAL;
  781. goto done;
  782. }
  783. if ((client_info_size > 0) && (client_info == NULL)) {
  784. pr_err("%s: User info pointer is NULL but size is %zd!\n",
  785. __func__, client_info_size);
  786. ret = -EINVAL;
  787. goto done;
  788. }
  789. if (alloc_data->cal_data.mem_handle < 0) {
  790. pr_err("%s: mem_handle %d invalid!\n",
  791. __func__, alloc_data->cal_data.mem_handle);
  792. ret = -EINVAL;
  793. goto done;
  794. }
  795. mutex_lock(&cal_type->lock);
  796. cal_block = get_matching_cal_block(cal_type,
  797. data);
  798. if (cal_block != NULL) {
  799. cal_block->cma_mem = alloc_data->cal_data.cma_mem;
  800. ret = unmap_memory(cal_type, cal_block);
  801. if (ret < 0)
  802. goto err;
  803. ret = realloc_memory(cal_block);
  804. if (ret < 0)
  805. goto err;
  806. } else {
  807. cal_block = create_cal_block(cal_type,
  808. (struct audio_cal_type_basic *)alloc_data,
  809. client_info_size, client_info);
  810. if (cal_block == NULL) {
  811. pr_err("%s: create_cal_block failed for %d!\n",
  812. __func__, alloc_data->cal_data.mem_handle);
  813. ret = -EINVAL;
  814. goto err;
  815. }
  816. }
  817. ret = map_memory(cal_type, cal_block);
  818. if (ret < 0)
  819. goto err;
  820. err:
  821. mutex_unlock(&cal_type->lock);
  822. done:
  823. return ret;
  824. }
  825. EXPORT_SYMBOL(cal_utils_alloc_cal);
  826. /**
  827. * cal_utils_dealloc_cal
  828. *
  829. * @data_size: size of data to allocate
  830. * @data: data pointer
  831. * @cal_type: pointer to the cal type
  832. *
  833. * Returns 0 on success, appropriate error code otherwise
  834. */
  835. int cal_utils_dealloc_cal(size_t data_size, void *data,
  836. struct cal_type_data *cal_type)
  837. {
  838. int ret = 0;
  839. struct cal_block_data *cal_block;
  840. struct audio_cal_type_dealloc *dealloc_data = data;
  841. unsigned long flags = 0;
  842. pr_debug("%s\n", __func__);
  843. if (cal_type == NULL) {
  844. pr_err("%s: cal_type is NULL!\n",
  845. __func__);
  846. ret = -EINVAL;
  847. goto done;
  848. }
  849. if (data_size < sizeof(struct audio_cal_type_dealloc)) {
  850. pr_err("%s: data_size of %zd does not equal struct size of %zd!\n",
  851. __func__, data_size,
  852. sizeof(struct audio_cal_type_dealloc));
  853. ret = -EINVAL;
  854. goto done;
  855. }
  856. if ((dealloc_data->cal_data.mem_handle == -1) &&
  857. (dealloc_data->cal_hdr.buffer_number == ALL_CAL_BLOCKS)) {
  858. destroy_all_cal_blocks(cal_type);
  859. goto done;
  860. }
  861. if (dealloc_data->cal_data.mem_handle < 0) {
  862. pr_err("%s: mem_handle %d invalid!\n",
  863. __func__, dealloc_data->cal_data.mem_handle);
  864. ret = -EINVAL;
  865. goto done;
  866. }
  867. mutex_lock(&cal_type->lock);
  868. cal_block = get_matching_cal_block(
  869. cal_type,
  870. data);
  871. if (cal_block == NULL) {
  872. pr_err("%s: allocation does not exist for %d!\n",
  873. __func__, dealloc_data->cal_data.mem_handle);
  874. ret = -EINVAL;
  875. goto err;
  876. }
  877. ret = unmap_memory(cal_type, cal_block);
  878. if (ret < 0)
  879. goto err;
  880. spin_lock_irqsave(&cal_lock, flags);
  881. delete_cal_block(cal_block);
  882. spin_unlock_irqrestore(&cal_lock, flags);
  883. err:
  884. mutex_unlock(&cal_type->lock);
  885. done:
  886. return ret;
  887. }
  888. EXPORT_SYMBOL(cal_utils_dealloc_cal);
  889. /**
  890. * cal_utils_set_cal
  891. *
  892. * @data_size: size of data to allocate
  893. * @data: data pointer
  894. * @cal_type: pointer to the cal type
  895. * @client_info_size: client info size
  896. * @client_info: pointer to client info
  897. *
  898. * Returns 0 on success, appropriate error code otherwise
  899. */
  900. int cal_utils_set_cal(size_t data_size, void *data,
  901. struct cal_type_data *cal_type,
  902. size_t client_info_size, void *client_info)
  903. {
  904. int ret = 0;
  905. struct cal_block_data *cal_block;
  906. struct audio_cal_type_basic *basic_data = data;
  907. pr_debug("%s\n", __func__);
  908. if (cal_type == NULL) {
  909. pr_err("%s: cal_type is NULL!\n",
  910. __func__);
  911. ret = -EINVAL;
  912. goto done;
  913. }
  914. if ((client_info_size > 0) && (client_info == NULL)) {
  915. pr_err("%s: User info pointer is NULL but size is %zd!\n",
  916. __func__, client_info_size);
  917. ret = -EINVAL;
  918. goto done;
  919. }
  920. if ((data_size > get_user_cal_type_size(
  921. cal_type->info.reg.cal_type)) || (data_size < 0)) {
  922. pr_err("%s: cal_type %d, data_size of %zd is invalid, expecting %zd!\n",
  923. __func__, cal_type->info.reg.cal_type, data_size,
  924. get_user_cal_type_size(cal_type->info.reg.cal_type));
  925. ret = -EINVAL;
  926. goto done;
  927. }
  928. mutex_lock(&cal_type->lock);
  929. cal_block = get_matching_cal_block(
  930. cal_type,
  931. data);
  932. if (cal_block == NULL) {
  933. if (basic_data->cal_data.mem_handle > 0) {
  934. pr_err("%s: allocation does not exist for %d!\n",
  935. __func__, basic_data->cal_data.mem_handle);
  936. ret = -EINVAL;
  937. goto err;
  938. } else {
  939. cal_block = create_cal_block(
  940. cal_type,
  941. basic_data,
  942. client_info_size, client_info);
  943. if (cal_block == NULL) {
  944. pr_err("%s: create_cal_block failed for cal type %d!\n",
  945. __func__,
  946. cal_type->info.reg.cal_type);
  947. ret = -EINVAL;
  948. goto err;
  949. }
  950. }
  951. }
  952. ret = map_memory(cal_type, cal_block);
  953. if (ret < 0)
  954. goto err;
  955. cal_block->cal_data.size = basic_data->cal_data.cal_size;
  956. if (client_info_size > 0) {
  957. memcpy(cal_block->client_info,
  958. client_info,
  959. client_info_size);
  960. }
  961. memcpy(cal_block->cal_info,
  962. ((uint8_t *)data + sizeof(struct audio_cal_type_basic)),
  963. data_size - sizeof(struct audio_cal_type_basic));
  964. /* reset buffer stale flag */
  965. cal_block->cal_stale = false;
  966. err:
  967. mutex_unlock(&cal_type->lock);
  968. done:
  969. return ret;
  970. }
  971. EXPORT_SYMBOL(cal_utils_set_cal);
  972. /**
  973. * cal_utils_mark_cal_used
  974. *
  975. * @cal_block: pointer to cal block
  976. */
  977. void cal_utils_mark_cal_used(struct cal_block_data *cal_block)
  978. {
  979. if (cal_block)
  980. cal_block->cal_stale = true;
  981. }
  982. EXPORT_SYMBOL(cal_utils_mark_cal_used);
  983. int __init cal_utils_init(void)
  984. {
  985. spin_lock_init(&cal_lock);
  986. return 0;
  987. }
  988. /**
  989. * cal_utils_is_cal_stale
  990. *
  991. * @cal_block: pointer to cal block
  992. *
  993. * @cal_type: pointer to the cal type
  994. *
  995. * Returns true if cal block is stale, false otherwise
  996. */
  997. bool cal_utils_is_cal_stale(struct cal_block_data *cal_block, struct cal_type_data *cal_type)
  998. {
  999. bool ret = false;
  1000. unsigned long flags = 0;
  1001. if (!cal_type) {
  1002. pr_err("%s: cal_type is Null", __func__);
  1003. goto done;
  1004. }
  1005. spin_lock_irqsave(&cal_lock, flags);
  1006. cal_block = cal_utils_get_only_cal_block(cal_type);
  1007. if (!cal_block) {
  1008. pr_err("%s: cal_block is Null", __func__);
  1009. goto unlock;
  1010. }
  1011. if (cal_block->cal_stale)
  1012. ret = true;
  1013. unlock:
  1014. spin_unlock_irqrestore(&cal_lock, flags);
  1015. done:
  1016. return ret;
  1017. }
  1018. EXPORT_SYMBOL(cal_utils_is_cal_stale);