audio_notifier.c 17 KB

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