audio_notifier.c 17 KB

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