audio_notifier.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2016-2017, The Linux Foundation. All rights reserved.
  4. */
  5. #include <linux/module.h>
  6. #include <linux/slab.h>
  7. #include <soc/qcom/subsystem_notif.h>
  8. #include <soc/qcom/service-notifier.h>
  9. #include <dsp/audio_notifier.h>
  10. #include "audio_ssr.h"
  11. #include "audio_pdr.h"
  12. /* Audio states internal to notifier. Client */
  13. /* used states defined in audio_notifier.h */
  14. /* for AUDIO_NOTIFIER_SERVICE_DOWN & UP */
  15. #define NO_SERVICE -2
  16. #define UNINIT_SERVICE -1
  17. /*
  18. * Used for each client registered with audio notifier
  19. */
  20. struct client_data {
  21. struct list_head list;
  22. /* Notifier block given by client */
  23. struct notifier_block *nb;
  24. char client_name[20];
  25. int service;
  26. int domain;
  27. };
  28. /*
  29. * Used for each service and domain combination
  30. * Tracks information specific to the underlying
  31. * service.
  32. */
  33. struct service_info {
  34. const char name[20];
  35. int domain_id;
  36. int state;
  37. void *handle;
  38. /* Notifier block registered to service */
  39. struct notifier_block *nb;
  40. /* Used to determine when to register and deregister service */
  41. int num_of_clients;
  42. /* List of all clients registered to the service and domain */
  43. struct srcu_notifier_head client_nb_list;
  44. };
  45. static int audio_notifer_ssr_adsp_cb(struct notifier_block *this,
  46. unsigned long opcode, void *data);
  47. static int audio_notifer_ssr_modem_cb(struct notifier_block *this,
  48. unsigned long opcode, void *data);
  49. static int audio_notifer_pdr_adsp_cb(struct notifier_block *this,
  50. unsigned long opcode, void *data);
  51. static struct notifier_block notifier_ssr_adsp_nb = {
  52. .notifier_call = audio_notifer_ssr_adsp_cb,
  53. .priority = 0,
  54. };
  55. static struct notifier_block notifier_ssr_modem_nb = {
  56. .notifier_call = audio_notifer_ssr_modem_cb,
  57. .priority = 0,
  58. };
  59. static struct notifier_block notifier_pdr_adsp_nb = {
  60. .notifier_call = audio_notifer_pdr_adsp_cb,
  61. .priority = 0,
  62. };
  63. static struct service_info service_data[AUDIO_NOTIFIER_MAX_SERVICES]
  64. [AUDIO_NOTIFIER_MAX_DOMAINS] = {
  65. {{
  66. .name = "SSR_ADSP",
  67. .domain_id = AUDIO_SSR_DOMAIN_ADSP,
  68. .state = AUDIO_NOTIFIER_SERVICE_DOWN,
  69. .nb = &notifier_ssr_adsp_nb
  70. },
  71. {
  72. .name = "SSR_MODEM",
  73. .domain_id = AUDIO_SSR_DOMAIN_MODEM,
  74. .state = AUDIO_NOTIFIER_SERVICE_DOWN,
  75. .nb = &notifier_ssr_modem_nb
  76. } },
  77. {{
  78. .name = "PDR_ADSP",
  79. .domain_id = AUDIO_PDR_DOMAIN_ADSP,
  80. .state = UNINIT_SERVICE,
  81. .nb = &notifier_pdr_adsp_nb
  82. },
  83. { /* PDR MODEM service not enabled */
  84. .name = "INVALID",
  85. .state = NO_SERVICE,
  86. .nb = NULL
  87. } }
  88. };
  89. /* Master list of all audio notifier clients */
  90. struct list_head client_list;
  91. struct mutex notifier_mutex;
  92. static int audio_notifer_get_default_service(int domain)
  93. {
  94. int service = NO_SERVICE;
  95. /* initial service to connect per domain */
  96. switch (domain) {
  97. case AUDIO_NOTIFIER_ADSP_DOMAIN:
  98. service = AUDIO_NOTIFIER_PDR_SERVICE;
  99. break;
  100. case AUDIO_NOTIFIER_MODEM_DOMAIN:
  101. service = AUDIO_NOTIFIER_SSR_SERVICE;
  102. break;
  103. }
  104. return service;
  105. }
  106. static void audio_notifer_disable_service(int service)
  107. {
  108. int i;
  109. for (i = 0; i < AUDIO_NOTIFIER_MAX_DOMAINS; i++)
  110. service_data[service][i].state = NO_SERVICE;
  111. }
  112. static bool audio_notifer_is_service_enabled(int service)
  113. {
  114. int i;
  115. for (i = 0; i < AUDIO_NOTIFIER_MAX_DOMAINS; i++)
  116. if (service_data[service][i].state != NO_SERVICE)
  117. return true;
  118. return false;
  119. }
  120. static void audio_notifer_init_service(int service)
  121. {
  122. int i;
  123. for (i = 0; i < AUDIO_NOTIFIER_MAX_DOMAINS; i++) {
  124. if (service_data[service][i].state == UNINIT_SERVICE)
  125. service_data[service][i].state =
  126. AUDIO_NOTIFIER_SERVICE_DOWN;
  127. }
  128. }
  129. static int audio_notifer_reg_service(int service, int domain)
  130. {
  131. void *handle;
  132. int ret = 0;
  133. int curr_state = AUDIO_NOTIFIER_SERVICE_DOWN;
  134. switch (service) {
  135. case AUDIO_NOTIFIER_SSR_SERVICE:
  136. handle = audio_ssr_register(
  137. service_data[service][domain].domain_id,
  138. service_data[service][domain].nb);
  139. break;
  140. case AUDIO_NOTIFIER_PDR_SERVICE:
  141. handle = audio_pdr_service_register(
  142. service_data[service][domain].domain_id,
  143. service_data[service][domain].nb, &curr_state);
  144. if (curr_state == SERVREG_NOTIF_SERVICE_STATE_UP_V01)
  145. curr_state = AUDIO_NOTIFIER_SERVICE_UP;
  146. else
  147. curr_state = AUDIO_NOTIFIER_SERVICE_DOWN;
  148. break;
  149. default:
  150. pr_err("%s: Invalid service %d\n",
  151. __func__, service);
  152. ret = -EINVAL;
  153. goto done;
  154. }
  155. if (IS_ERR_OR_NULL(handle)) {
  156. pr_err("%s: handle is incorrect for service %s\n",
  157. __func__, service_data[service][domain].name);
  158. ret = -EINVAL;
  159. goto done;
  160. }
  161. service_data[service][domain].state = curr_state;
  162. service_data[service][domain].handle = handle;
  163. pr_info("%s: service %s is in use\n",
  164. __func__, service_data[service][domain].name);
  165. pr_debug("%s: service %s has current state %d, handle 0x%pK\n",
  166. __func__, service_data[service][domain].name,
  167. service_data[service][domain].state,
  168. service_data[service][domain].handle);
  169. done:
  170. return ret;
  171. }
  172. static int audio_notifer_dereg_service(int service, int domain)
  173. {
  174. int ret;
  175. switch (service) {
  176. case AUDIO_NOTIFIER_SSR_SERVICE:
  177. ret = audio_ssr_deregister(
  178. service_data[service][domain].handle,
  179. service_data[service][domain].nb);
  180. break;
  181. case AUDIO_NOTIFIER_PDR_SERVICE:
  182. ret = audio_pdr_service_deregister(
  183. service_data[service][domain].handle,
  184. service_data[service][domain].nb);
  185. break;
  186. default:
  187. pr_err("%s: Invalid service %d\n",
  188. __func__, service);
  189. ret = -EINVAL;
  190. goto done;
  191. }
  192. if (ret < 0) {
  193. pr_err("%s: deregister failed for service %s, ret %d\n",
  194. __func__, service_data[service][domain].name, ret);
  195. goto done;
  196. }
  197. pr_debug("%s: service %s with handle 0x%pK deregistered\n",
  198. __func__, service_data[service][domain].name,
  199. service_data[service][domain].handle);
  200. service_data[service][domain].state = AUDIO_NOTIFIER_SERVICE_DOWN;
  201. service_data[service][domain].handle = NULL;
  202. done:
  203. return ret;
  204. }
  205. static int audio_notifer_reg_client_service(struct client_data *client_data,
  206. int service)
  207. {
  208. int ret = 0;
  209. int domain = client_data->domain;
  210. struct audio_notifier_cb_data data;
  211. switch (service) {
  212. case AUDIO_NOTIFIER_SSR_SERVICE:
  213. case AUDIO_NOTIFIER_PDR_SERVICE:
  214. if (service_data[service][domain].num_of_clients == 0)
  215. ret = audio_notifer_reg_service(service, domain);
  216. break;
  217. default:
  218. pr_err("%s: Invalid service for client %s, service %d, domain %d\n",
  219. __func__, client_data->client_name, service, domain);
  220. ret = -EINVAL;
  221. goto done;
  222. }
  223. if (ret < 0) {
  224. pr_err("%s: service registration failed on service %s for client %s\n",
  225. __func__, service_data[service][domain].name,
  226. client_data->client_name);
  227. goto done;
  228. }
  229. client_data->service = service;
  230. srcu_notifier_chain_register(
  231. &service_data[service][domain].client_nb_list,
  232. client_data->nb);
  233. service_data[service][domain].num_of_clients++;
  234. pr_debug("%s: registered client %s on service %s, current state 0x%x\n",
  235. __func__, client_data->client_name,
  236. service_data[service][domain].name,
  237. service_data[service][domain].state);
  238. /*
  239. * PDR registration returns current state
  240. * Force callback of client with current state for PDR
  241. */
  242. if (client_data->service == AUDIO_NOTIFIER_PDR_SERVICE) {
  243. data.service = service;
  244. data.domain = domain;
  245. (void)client_data->nb->notifier_call(client_data->nb,
  246. service_data[service][domain].state, &data);
  247. }
  248. done:
  249. return ret;
  250. }
  251. static int audio_notifer_reg_client(struct client_data *client_data)
  252. {
  253. int ret = 0;
  254. int service;
  255. int domain = client_data->domain;
  256. service = audio_notifer_get_default_service(domain);
  257. if (service < 0) {
  258. pr_err("%s: service %d is incorrect\n", __func__, service);
  259. ret = -EINVAL;
  260. goto done;
  261. }
  262. /* Search through services to find a valid one to register client on. */
  263. for (; service >= 0; service--) {
  264. /* If a service is not initialized, wait for it to come up. */
  265. if (service_data[service][domain].state == UNINIT_SERVICE)
  266. goto done;
  267. /* Skip unsupported service and domain combinations. */
  268. if (service_data[service][domain].state < 0)
  269. continue;
  270. /* Only register clients who have not acquired a service. */
  271. if (client_data->service != NO_SERVICE)
  272. continue;
  273. /*
  274. * Only register clients, who have not acquired a service, on
  275. * the best available service for their domain. Uninitialized
  276. * services will try to register all of their clients after
  277. * they initialize correctly or will disable their service and
  278. * register clients on the next best avaialable service.
  279. */
  280. pr_debug("%s: register client %s on service %s",
  281. __func__, client_data->client_name,
  282. service_data[service][domain].name);
  283. ret = audio_notifer_reg_client_service(client_data, service);
  284. if (ret < 0)
  285. pr_err("%s: client %s failed to register on service %s",
  286. __func__, client_data->client_name,
  287. service_data[service][domain].name);
  288. }
  289. done:
  290. return ret;
  291. }
  292. static int audio_notifer_dereg_client(struct client_data *client_data)
  293. {
  294. int ret = 0;
  295. int service = client_data->service;
  296. int domain = client_data->domain;
  297. switch (client_data->service) {
  298. case AUDIO_NOTIFIER_SSR_SERVICE:
  299. case AUDIO_NOTIFIER_PDR_SERVICE:
  300. if (service_data[service][domain].num_of_clients == 1)
  301. ret = audio_notifer_dereg_service(service, domain);
  302. break;
  303. case NO_SERVICE:
  304. goto done;
  305. default:
  306. pr_err("%s: Invalid service for client %s, service %d\n",
  307. __func__, client_data->client_name,
  308. client_data->service);
  309. ret = -EINVAL;
  310. goto done;
  311. }
  312. if (ret < 0) {
  313. pr_err("%s: deregister failed for client %s on service %s, ret %d\n",
  314. __func__, client_data->client_name,
  315. service_data[service][domain].name, ret);
  316. goto done;
  317. }
  318. ret = srcu_notifier_chain_unregister(&service_data[service][domain].
  319. client_nb_list, client_data->nb);
  320. if (ret < 0) {
  321. pr_err("%s: srcu_notifier_chain_unregister failed, ret %d\n",
  322. __func__, ret);
  323. goto done;
  324. }
  325. pr_debug("%s: deregistered client %s on service %s\n",
  326. __func__, client_data->client_name,
  327. service_data[service][domain].name);
  328. client_data->service = NO_SERVICE;
  329. if (service_data[service][domain].num_of_clients > 0)
  330. service_data[service][domain].num_of_clients--;
  331. done:
  332. return ret;
  333. }
  334. static void audio_notifer_reg_all_clients(void)
  335. {
  336. struct list_head *ptr, *next;
  337. struct client_data *client_data;
  338. int ret;
  339. list_for_each_safe(ptr, next, &client_list) {
  340. client_data = list_entry(ptr, struct client_data, list);
  341. ret = audio_notifer_reg_client(client_data);
  342. if (ret < 0)
  343. pr_err("%s: audio_notifer_reg_client failed for client %s, ret %d\n",
  344. __func__, client_data->client_name,
  345. ret);
  346. }
  347. }
  348. static int audio_notifer_pdr_callback(struct notifier_block *this,
  349. unsigned long opcode, void *data)
  350. {
  351. pr_debug("%s: Audio PDR framework state 0x%lx\n",
  352. __func__, opcode);
  353. mutex_lock(&notifier_mutex);
  354. if (opcode == AUDIO_PDR_FRAMEWORK_DOWN)
  355. audio_notifer_disable_service(AUDIO_NOTIFIER_PDR_SERVICE);
  356. else
  357. audio_notifer_init_service(AUDIO_NOTIFIER_PDR_SERVICE);
  358. audio_notifer_reg_all_clients();
  359. mutex_unlock(&notifier_mutex);
  360. return 0;
  361. }
  362. static struct notifier_block pdr_nb = {
  363. .notifier_call = audio_notifer_pdr_callback,
  364. .priority = 0,
  365. };
  366. static int audio_notifer_convert_opcode(unsigned long opcode,
  367. unsigned long *notifier_opcode)
  368. {
  369. int ret = 0;
  370. switch (opcode) {
  371. case SUBSYS_BEFORE_SHUTDOWN:
  372. case SERVREG_NOTIF_SERVICE_STATE_DOWN_V01:
  373. *notifier_opcode = AUDIO_NOTIFIER_SERVICE_DOWN;
  374. break;
  375. case SUBSYS_AFTER_POWERUP:
  376. case SERVREG_NOTIF_SERVICE_STATE_UP_V01:
  377. *notifier_opcode = AUDIO_NOTIFIER_SERVICE_UP;
  378. break;
  379. default:
  380. pr_debug("%s: Unused opcode 0x%lx\n", __func__, opcode);
  381. ret = -EINVAL;
  382. }
  383. return ret;
  384. }
  385. static int audio_notifer_service_cb(unsigned long opcode,
  386. int service, int domain)
  387. {
  388. int ret = 0;
  389. unsigned long notifier_opcode;
  390. struct audio_notifier_cb_data data;
  391. if (audio_notifer_convert_opcode(opcode, &notifier_opcode) < 0)
  392. goto done;
  393. data.service = service;
  394. data.domain = domain;
  395. pr_debug("%s: service %s, opcode 0x%lx\n",
  396. __func__, service_data[service][domain].name, notifier_opcode);
  397. mutex_lock(&notifier_mutex);
  398. service_data[service][domain].state = notifier_opcode;
  399. ret = srcu_notifier_call_chain(&service_data[service][domain].
  400. client_nb_list, notifier_opcode, &data);
  401. if (ret < 0)
  402. pr_err("%s: srcu_notifier_call_chain returned %d, service %s, opcode 0x%lx\n",
  403. __func__, ret, service_data[service][domain].name,
  404. notifier_opcode);
  405. mutex_unlock(&notifier_mutex);
  406. done:
  407. return NOTIFY_OK;
  408. }
  409. static int audio_notifer_pdr_adsp_cb(struct notifier_block *this,
  410. unsigned long opcode, void *data)
  411. {
  412. return audio_notifer_service_cb(opcode,
  413. AUDIO_NOTIFIER_PDR_SERVICE,
  414. AUDIO_NOTIFIER_ADSP_DOMAIN);
  415. }
  416. static int audio_notifer_ssr_adsp_cb(struct notifier_block *this,
  417. unsigned long opcode, void *data)
  418. {
  419. return audio_notifer_service_cb(opcode,
  420. AUDIO_NOTIFIER_SSR_SERVICE,
  421. AUDIO_NOTIFIER_ADSP_DOMAIN);
  422. }
  423. static int audio_notifer_ssr_modem_cb(struct notifier_block *this,
  424. unsigned long opcode, void *data)
  425. {
  426. return audio_notifer_service_cb(opcode,
  427. AUDIO_NOTIFIER_SSR_SERVICE,
  428. AUDIO_NOTIFIER_MODEM_DOMAIN);
  429. }
  430. int audio_notifier_deregister(char *client_name)
  431. {
  432. int ret = 0;
  433. int ret2;
  434. struct list_head *ptr, *next;
  435. struct client_data *client_data = NULL;
  436. if (client_name == NULL) {
  437. pr_err("%s: client_name is NULL\n", __func__);
  438. ret = -EINVAL;
  439. goto done;
  440. }
  441. mutex_lock(&notifier_mutex);
  442. list_for_each_safe(ptr, next, &client_list) {
  443. client_data = list_entry(ptr, struct client_data, list);
  444. if (!strcmp(client_name, client_data->client_name)) {
  445. ret2 = audio_notifer_dereg_client(client_data);
  446. if (ret2 < 0) {
  447. pr_err("%s: audio_notifer_dereg_client failed, ret %d\n, service %d, domain %d",
  448. __func__, ret2, client_data->service,
  449. client_data->domain);
  450. ret = ret2;
  451. continue;
  452. }
  453. list_del(&client_data->list);
  454. kfree(client_data);
  455. }
  456. }
  457. mutex_unlock(&notifier_mutex);
  458. done:
  459. return ret;
  460. }
  461. EXPORT_SYMBOL(audio_notifier_deregister);
  462. int audio_notifier_register(char *client_name, int domain,
  463. struct notifier_block *nb)
  464. {
  465. int ret;
  466. struct client_data *client_data;
  467. if (client_name == NULL) {
  468. pr_err("%s: client_name is NULL\n", __func__);
  469. ret = -EINVAL;
  470. goto done;
  471. } else if (nb == NULL) {
  472. pr_err("%s: Notifier block is NULL\n", __func__);
  473. ret = -EINVAL;
  474. goto done;
  475. }
  476. client_data = kmalloc(sizeof(*client_data), GFP_KERNEL);
  477. if (client_data == NULL) {
  478. ret = -ENOMEM;
  479. goto done;
  480. }
  481. INIT_LIST_HEAD(&client_data->list);
  482. client_data->nb = nb;
  483. strlcpy(client_data->client_name, client_name,
  484. sizeof(client_data->client_name));
  485. client_data->service = NO_SERVICE;
  486. client_data->domain = domain;
  487. mutex_lock(&notifier_mutex);
  488. ret = audio_notifer_reg_client(client_data);
  489. if (ret < 0) {
  490. mutex_unlock(&notifier_mutex);
  491. pr_err("%s: audio_notifer_reg_client for client %s failed ret = %d\n",
  492. __func__, client_data->client_name,
  493. ret);
  494. kfree(client_data);
  495. goto done;
  496. }
  497. list_add_tail(&client_data->list, &client_list);
  498. mutex_unlock(&notifier_mutex);
  499. done:
  500. return ret;
  501. }
  502. EXPORT_SYMBOL(audio_notifier_register);
  503. static int __init audio_notifier_subsys_init(void)
  504. {
  505. int i, j;
  506. mutex_init(&notifier_mutex);
  507. INIT_LIST_HEAD(&client_list);
  508. for (i = 0; i < AUDIO_NOTIFIER_MAX_SERVICES; i++) {
  509. for (j = 0; j < AUDIO_NOTIFIER_MAX_DOMAINS; j++) {
  510. if (service_data[i][j].state <= NO_SERVICE)
  511. continue;
  512. srcu_init_notifier_head(
  513. &service_data[i][j].client_nb_list);
  514. }
  515. }
  516. return 0;
  517. }
  518. static int __init audio_notifier_late_init(void)
  519. {
  520. /*
  521. * If pdr registration failed, register clients on next service
  522. * Do in late init to ensure that SSR subsystem is initialized
  523. */
  524. mutex_lock(&notifier_mutex);
  525. if (!audio_notifer_is_service_enabled(AUDIO_NOTIFIER_PDR_SERVICE))
  526. audio_notifer_reg_all_clients();
  527. mutex_unlock(&notifier_mutex);
  528. return 0;
  529. }
  530. static int __init audio_notifier_init(void)
  531. {
  532. int ret;
  533. audio_notifier_subsys_init();
  534. ret = audio_pdr_register(&pdr_nb);
  535. if (ret < 0) {
  536. pr_err("%s: PDR register failed, ret = %d, disable service\n",
  537. __func__, ret);
  538. audio_notifer_disable_service(AUDIO_NOTIFIER_PDR_SERVICE);
  539. }
  540. /* Do not return error since PDR enablement is not critical */
  541. audio_notifier_late_init();
  542. return 0;
  543. }
  544. module_init(audio_notifier_init);
  545. static void __exit audio_notifier_exit(void)
  546. {
  547. audio_pdr_deregister(&pdr_nb);
  548. }
  549. module_exit(audio_notifier_exit);
  550. MODULE_DESCRIPTION("Audio notifier driver");
  551. MODULE_LICENSE("GPL v2");