audio_notifier.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Copyright (c) 2016-2017, 2020-2021 The Linux Foundation. All rights reserved.
  4. * Copyright (c) 2022 Qualcomm Innovation Center, Inc. All rights reserved.
  5. */
  6. #include <linux/init.h>
  7. #include <linux/kernel.h>
  8. #include <linux/module.h>
  9. #include <linux/err.h>
  10. #include <linux/string.h>
  11. #include <linux/delay.h>
  12. #include <linux/platform_device.h>
  13. #include <linux/of_device.h>
  14. #include <linux/slab.h>
  15. #include <linux/remoteproc.h>
  16. #include <linux/remoteproc/qcom_rproc.h>
  17. #include <dsp/audio_notifier.h>
  18. #include "audio_ssr.h"
  19. #include "audio_pdr.h"
  20. #include <trace/events/rproc_qcom.h>
  21. /* Audio states internal to notifier. Client */
  22. /* used states defined in audio_notifier.h */
  23. /* for AUDIO_NOTIFIER_SERVICE_DOWN & UP */
  24. #define NO_SERVICE -2
  25. #define UNINIT_SERVICE -1
  26. static struct platform_device *adsp_private;
  27. struct adsp_notify_private {
  28. struct rproc *rproc_h;
  29. bool notifier_probe_complete;
  30. };
  31. /*
  32. * Used for each client registered with audio notifier
  33. */
  34. struct client_data {
  35. struct list_head list;
  36. /* Notifier block given by client */
  37. struct notifier_block *nb;
  38. char client_name[20];
  39. int service;
  40. int domain;
  41. };
  42. /*
  43. * Used for each service and domain combination
  44. * Tracks information specific to the underlying
  45. * service.
  46. */
  47. struct service_info {
  48. const char name[20];
  49. int domain_id;
  50. int state;
  51. void *handle;
  52. /* Hook registered to service */
  53. union {
  54. void (*cb)(int, char *, void *);
  55. struct notifier_block *nb;
  56. } hook;
  57. /* Used to determine when to register and deregister service */
  58. int num_of_clients;
  59. /* List of all clients registered to the service and domain */
  60. struct srcu_notifier_head client_nb_list;
  61. };
  62. static int audio_notifier_ssr_adsp_cb(struct notifier_block *this,
  63. unsigned long opcode, void *data);
  64. static int audio_notifier_ssr_modem_cb(struct notifier_block *this,
  65. unsigned long opcode, void *data);
  66. static void audio_notifier_pdr_adsp_cb(int status, char *service_name, void *priv);
  67. static struct notifier_block notifier_ssr_adsp_nb = {
  68. .notifier_call = audio_notifier_ssr_adsp_cb,
  69. .priority = 0,
  70. };
  71. static struct notifier_block notifier_ssr_modem_nb = {
  72. .notifier_call = audio_notifier_ssr_modem_cb,
  73. .priority = 0,
  74. };
  75. static struct service_info service_data[AUDIO_NOTIFIER_MAX_SERVICES]
  76. [AUDIO_NOTIFIER_MAX_DOMAINS] = {
  77. {{
  78. .name = "SSR_ADSP",
  79. .domain_id = AUDIO_SSR_DOMAIN_ADSP,
  80. .state = AUDIO_NOTIFIER_SERVICE_DOWN,
  81. .hook.nb = &notifier_ssr_adsp_nb
  82. },
  83. {
  84. .name = "SSR_MODEM",
  85. .domain_id = AUDIO_SSR_DOMAIN_MODEM,
  86. .state = AUDIO_NOTIFIER_SERVICE_DOWN,
  87. .hook.nb = &notifier_ssr_modem_nb
  88. } },
  89. {{
  90. .name = "PDR_ADSP",
  91. .domain_id = AUDIO_PDR_DOMAIN_ADSP,
  92. .state = UNINIT_SERVICE,
  93. .hook.cb = &audio_notifier_pdr_adsp_cb
  94. },
  95. { /* PDR MODEM service not enabled */
  96. .name = "INVALID",
  97. .state = NO_SERVICE,
  98. .hook.nb = NULL
  99. } }
  100. };
  101. /* Master list of all audio notifier clients */
  102. LIST_HEAD(client_list);
  103. struct mutex notifier_mutex;
  104. static int audio_notifier_get_default_service(int domain)
  105. {
  106. int service = NO_SERVICE;
  107. /* initial service to connect per domain */
  108. switch (domain) {
  109. case AUDIO_NOTIFIER_ADSP_DOMAIN:
  110. service = AUDIO_NOTIFIER_PDR_SERVICE;
  111. break;
  112. case AUDIO_NOTIFIER_MODEM_DOMAIN:
  113. service = AUDIO_NOTIFIER_SSR_SERVICE;
  114. break;
  115. }
  116. return service;
  117. }
  118. #ifdef CONFIG_MSM_QDSP6_PDR
  119. static void audio_notifier_init_service(int service)
  120. {
  121. int i;
  122. for (i = 0; i < AUDIO_NOTIFIER_MAX_DOMAINS; i++) {
  123. if (service_data[service][i].state == UNINIT_SERVICE)
  124. service_data[service][i].state =
  125. AUDIO_NOTIFIER_SERVICE_DOWN;
  126. }
  127. }
  128. #else
  129. static void audio_notifier_init_service(int service)
  130. {
  131. int i;
  132. for (i = 0; i < AUDIO_NOTIFIER_MAX_DOMAINS; i++)
  133. service_data[service][i].state = NO_SERVICE;
  134. }
  135. #endif
  136. static bool audio_notifier_is_service_enabled(int service)
  137. {
  138. int i;
  139. for (i = 0; i < AUDIO_NOTIFIER_MAX_DOMAINS; i++)
  140. if (service_data[service][i].state != NO_SERVICE)
  141. return true;
  142. return false;
  143. }
  144. static int audio_notifier_reg_service(int service, int domain)
  145. {
  146. void *handle;
  147. int ret = 0;
  148. int curr_state = AUDIO_NOTIFIER_SERVICE_DOWN;
  149. struct platform_device *pdev = adsp_private;
  150. struct adsp_notify_private *priv = NULL;
  151. struct rproc *rproc;
  152. priv = platform_get_drvdata(pdev);
  153. if (!priv) {
  154. dev_err_ratelimited(&pdev->dev, " %s: Private data get failed\n", __func__);
  155. return ret;;
  156. }
  157. rproc = priv->rproc_h;
  158. switch (service) {
  159. case AUDIO_NOTIFIER_SSR_SERVICE:
  160. handle = audio_ssr_register(rproc->name,
  161. service_data[service][domain].hook.nb);
  162. break;
  163. case AUDIO_NOTIFIER_PDR_SERVICE:
  164. handle = audio_pdr_service_register(
  165. service_data[service][domain].domain_id,
  166. service_data[service][domain].hook.cb);
  167. curr_state = AUDIO_NOTIFIER_SERVICE_DOWN;
  168. break;
  169. default:
  170. pr_err_ratelimited("%s: Invalid service %d\n",
  171. __func__, service);
  172. ret = -EINVAL;
  173. goto done;
  174. }
  175. if (IS_ERR_OR_NULL(handle)) {
  176. pr_err_ratelimited("%s: handle is incorrect for service %s\n",
  177. __func__, service_data[service][domain].name);
  178. ret = -EINVAL;
  179. goto done;
  180. }
  181. service_data[service][domain].state = curr_state;
  182. service_data[service][domain].handle = handle;
  183. pr_info("%s: service %s is in use\n",
  184. __func__, service_data[service][domain].name);
  185. pr_debug("%s: service %s has current state %d, handle 0x%pK\n",
  186. __func__, service_data[service][domain].name,
  187. service_data[service][domain].state,
  188. service_data[service][domain].handle);
  189. done:
  190. return ret;
  191. }
  192. static int audio_notifier_dereg_service(int service, int domain)
  193. {
  194. int ret;
  195. switch (service) {
  196. case AUDIO_NOTIFIER_SSR_SERVICE:
  197. ret = audio_ssr_deregister(
  198. service_data[service][domain].handle,
  199. service_data[service][domain].hook.nb);
  200. break;
  201. case AUDIO_NOTIFIER_PDR_SERVICE:
  202. ret = audio_pdr_service_deregister(
  203. service_data[service][domain].domain_id);
  204. break;
  205. default:
  206. pr_err_ratelimited("%s: Invalid service %d\n",
  207. __func__, service);
  208. ret = -EINVAL;
  209. goto done;
  210. }
  211. if (ret < 0) {
  212. pr_err_ratelimited("%s: deregister failed for service %s, ret %d\n",
  213. __func__, service_data[service][domain].name, ret);
  214. goto done;
  215. }
  216. pr_debug("%s: service %s with handle 0x%pK deregistered\n",
  217. __func__, service_data[service][domain].name,
  218. service_data[service][domain].handle);
  219. service_data[service][domain].state = AUDIO_NOTIFIER_SERVICE_DOWN;
  220. service_data[service][domain].handle = NULL;
  221. done:
  222. return ret;
  223. }
  224. static int audio_notifier_reg_client_service(struct client_data *client_data,
  225. int service)
  226. {
  227. int ret = 0;
  228. int domain = client_data->domain;
  229. struct audio_notifier_cb_data data;
  230. switch (service) {
  231. case AUDIO_NOTIFIER_SSR_SERVICE:
  232. case AUDIO_NOTIFIER_PDR_SERVICE:
  233. if (service_data[service][domain].num_of_clients == 0)
  234. ret = audio_notifier_reg_service(service, domain);
  235. break;
  236. default:
  237. pr_err_ratelimited("%s: Invalid service for client %s, service %d, domain %d\n",
  238. __func__, client_data->client_name, service, domain);
  239. ret = -EINVAL;
  240. goto done;
  241. }
  242. if (ret < 0) {
  243. pr_err_ratelimited("%s: service registration failed on service %s for client %s\n",
  244. __func__, service_data[service][domain].name,
  245. client_data->client_name);
  246. goto done;
  247. }
  248. client_data->service = service;
  249. srcu_notifier_chain_register(
  250. &service_data[service][domain].client_nb_list,
  251. client_data->nb);
  252. service_data[service][domain].num_of_clients++;
  253. pr_debug("%s: registered client %s on service %s, current state 0x%x\n",
  254. __func__, client_data->client_name,
  255. service_data[service][domain].name,
  256. service_data[service][domain].state);
  257. /*
  258. * PDR registration returns current state
  259. * Force callback of client with current state for PDR
  260. */
  261. if (client_data->service == AUDIO_NOTIFIER_PDR_SERVICE) {
  262. data.service = service;
  263. data.domain = domain;
  264. (void)client_data->nb->notifier_call(client_data->nb,
  265. service_data[service][domain].state, &data);
  266. }
  267. done:
  268. return ret;
  269. }
  270. static int audio_notifier_reg_client(struct client_data *client_data)
  271. {
  272. int ret = 0;
  273. int service;
  274. int domain = client_data->domain;
  275. service = audio_notifier_get_default_service(domain);
  276. if (service < 0) {
  277. pr_err_ratelimited("%s: service %d is incorrect\n", __func__, service);
  278. ret = -EINVAL;
  279. goto done;
  280. }
  281. /* Search through services to find a valid one to register client on. */
  282. for (; service >= 0; service--) {
  283. /* If a service is not initialized, wait for it to come up. */
  284. if (service_data[service][domain].state == UNINIT_SERVICE) {
  285. pr_err_ratelimited("%s: failed in client registration to PDR\n",
  286. __func__);
  287. ret = -EINVAL;
  288. goto done;
  289. }
  290. /* Skip unsupported service and domain combinations. */
  291. if (service_data[service][domain].state < 0)
  292. continue;
  293. /* Only register clients who have not acquired a service. */
  294. if (client_data->service != NO_SERVICE)
  295. continue;
  296. /*
  297. * Only register clients, who have not acquired a service, on
  298. * the best available service for their domain. Uninitialized
  299. * services will try to register all of their clients after
  300. * they initialize correctly or will disable their service and
  301. * register clients on the next best avaialable service.
  302. */
  303. pr_debug("%s: register client %s on service %s",
  304. __func__, client_data->client_name,
  305. service_data[service][domain].name);
  306. ret = audio_notifier_reg_client_service(client_data, service);
  307. if (ret < 0)
  308. pr_err_ratelimited("%s: client %s failed to register on service %s",
  309. __func__, client_data->client_name,
  310. service_data[service][domain].name);
  311. }
  312. done:
  313. return ret;
  314. }
  315. static int audio_notifier_dereg_client(struct client_data *client_data)
  316. {
  317. int ret = 0;
  318. int service = client_data->service;
  319. int domain = client_data->domain;
  320. switch (client_data->service) {
  321. case AUDIO_NOTIFIER_SSR_SERVICE:
  322. case AUDIO_NOTIFIER_PDR_SERVICE:
  323. if (service_data[service][domain].num_of_clients == 1)
  324. ret = audio_notifier_dereg_service(service, domain);
  325. break;
  326. case NO_SERVICE:
  327. goto done;
  328. default:
  329. pr_err_ratelimited("%s: Invalid service for client %s, service %d\n",
  330. __func__, client_data->client_name,
  331. client_data->service);
  332. ret = -EINVAL;
  333. goto done;
  334. }
  335. if (ret < 0) {
  336. pr_err_ratelimited("%s: deregister failed for client %s on service %s, ret %d\n",
  337. __func__, client_data->client_name,
  338. service_data[service][domain].name, ret);
  339. goto done;
  340. }
  341. ret = srcu_notifier_chain_unregister(&service_data[service][domain].
  342. client_nb_list, client_data->nb);
  343. if (ret < 0) {
  344. pr_err_ratelimited("%s: srcu_notifier_chain_unregister failed, ret %d\n",
  345. __func__, ret);
  346. goto done;
  347. }
  348. pr_debug("%s: deregistered client %s on service %s\n",
  349. __func__, client_data->client_name,
  350. service_data[service][domain].name);
  351. client_data->service = NO_SERVICE;
  352. if (service_data[service][domain].num_of_clients > 0)
  353. service_data[service][domain].num_of_clients--;
  354. done:
  355. return ret;
  356. }
  357. static void audio_notifier_reg_all_clients(void)
  358. {
  359. struct list_head *ptr, *next;
  360. struct client_data *client_data;
  361. int ret;
  362. list_for_each_safe(ptr, next, &client_list) {
  363. client_data = list_entry(ptr, struct client_data, list);
  364. ret = audio_notifier_reg_client(client_data);
  365. if (ret < 0)
  366. pr_err_ratelimited("%s: audio_notifier_reg_client failed for client %s, \
  367. ret %d\n", __func__, client_data->client_name,
  368. ret);
  369. }
  370. }
  371. static int audio_notifier_convert_opcode(unsigned long opcode,
  372. unsigned long *notifier_opcode)
  373. {
  374. int ret = 0;
  375. switch (opcode) {
  376. case QCOM_SSR_BEFORE_SHUTDOWN:
  377. case SERVREG_SERVICE_STATE_DOWN:
  378. trace_rproc_qcom_event("audio",
  379. "QCOM_SSR_BEFORE_SHUTDOWN", "audio_notifier_convert_opcode_enter");
  380. *notifier_opcode = AUDIO_NOTIFIER_SERVICE_DOWN;
  381. break;
  382. case QCOM_SSR_AFTER_POWERUP:
  383. case SERVREG_SERVICE_STATE_UP:
  384. trace_rproc_qcom_event(
  385. "audio", "QCOM_SSR_AFTER_POWERUP", "audio_notifier_convert_opcode_enter");
  386. *notifier_opcode = AUDIO_NOTIFIER_SERVICE_UP;
  387. break;
  388. default:
  389. pr_debug("%s: Unused opcode 0x%lx\n", __func__, opcode);
  390. ret = -EINVAL;
  391. }
  392. return ret;
  393. }
  394. static int audio_notifier_service_cb(unsigned long opcode,
  395. int service, int domain)
  396. {
  397. int ret = 0;
  398. unsigned long notifier_opcode;
  399. struct audio_notifier_cb_data data;
  400. if (audio_notifier_convert_opcode(opcode, &notifier_opcode) < 0)
  401. goto done;
  402. data.service = service;
  403. data.domain = domain;
  404. pr_info("%s: service %s, opcode 0x%lx\n",
  405. __func__, service_data[service][domain].name, notifier_opcode);
  406. mutex_lock(&notifier_mutex);
  407. service_data[service][domain].state = notifier_opcode;
  408. ret = srcu_notifier_call_chain(&service_data[service][domain].
  409. client_nb_list, notifier_opcode, &data);
  410. if (ret < 0)
  411. pr_err_ratelimited("%s: srcu_notifier_call_chain returned %d, service %s, \
  412. opcode 0x%lx\n", __func__, ret, service_data[service][domain].name,
  413. notifier_opcode);
  414. mutex_unlock(&notifier_mutex);
  415. done:
  416. trace_rproc_qcom_event("audio", "audio_notifier", "audio_notifier_service_cb_exit");
  417. return NOTIFY_OK;
  418. }
  419. static void audio_notifier_pdr_adsp_cb(int status, char *service_name, void *priv)
  420. {
  421. audio_notifier_service_cb(status, AUDIO_NOTIFIER_PDR_SERVICE, AUDIO_NOTIFIER_ADSP_DOMAIN);
  422. }
  423. static int audio_notifier_ssr_adsp_cb(struct notifier_block *this,
  424. unsigned long opcode, void *data)
  425. {
  426. return audio_notifier_service_cb(opcode,
  427. AUDIO_NOTIFIER_SSR_SERVICE,
  428. AUDIO_NOTIFIER_ADSP_DOMAIN);
  429. }
  430. static int audio_notifier_ssr_modem_cb(struct notifier_block *this,
  431. unsigned long opcode, void *data)
  432. {
  433. return audio_notifier_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_ratelimited("%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_notifier_dereg_client(client_data);
  453. if (ret2 < 0) {
  454. pr_err_ratelimited("%s: audio_notifier_dereg_client failed, \
  455. ret %d\n, service %d, domain %d",
  456. __func__, ret2, client_data->service,
  457. client_data->domain);
  458. ret = ret2;
  459. continue;
  460. }
  461. list_del(&client_data->list);
  462. kfree(client_data);
  463. }
  464. }
  465. mutex_unlock(&notifier_mutex);
  466. done:
  467. return ret;
  468. }
  469. EXPORT_SYMBOL(audio_notifier_deregister);
  470. int audio_notifier_register(char *client_name, int domain,
  471. struct notifier_block *nb)
  472. {
  473. int ret;
  474. struct client_data *client_data;
  475. if (client_name == NULL) {
  476. pr_err_ratelimited("%s: client_name is NULL\n", __func__);
  477. ret = -EINVAL;
  478. goto done;
  479. } else if (nb == NULL) {
  480. pr_err_ratelimited("%s: Notifier block is NULL\n", __func__);
  481. ret = -EINVAL;
  482. goto done;
  483. }
  484. client_data = kmalloc(sizeof(*client_data), GFP_KERNEL);
  485. if (client_data == NULL) {
  486. ret = -ENOMEM;
  487. goto done;
  488. }
  489. INIT_LIST_HEAD(&client_data->list);
  490. client_data->nb = nb;
  491. strlcpy(client_data->client_name, client_name,
  492. sizeof(client_data->client_name));
  493. client_data->service = NO_SERVICE;
  494. client_data->domain = domain;
  495. mutex_lock(&notifier_mutex);
  496. ret = audio_notifier_reg_client(client_data);
  497. if (ret < 0) {
  498. mutex_unlock(&notifier_mutex);
  499. pr_err_ratelimited("%s: audio_notifier_reg_client for client %s failed ret = %d\n",
  500. __func__, client_data->client_name,
  501. ret);
  502. kfree(client_data);
  503. goto done;
  504. }
  505. list_add_tail(&client_data->list, &client_list);
  506. mutex_unlock(&notifier_mutex);
  507. done:
  508. return ret;
  509. }
  510. EXPORT_SYMBOL(audio_notifier_register);
  511. static int audio_notifier_subsys_init(void)
  512. {
  513. int i, j;
  514. mutex_init(&notifier_mutex);
  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 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_notifier_is_service_enabled(AUDIO_NOTIFIER_PDR_SERVICE))
  533. audio_notifier_reg_all_clients();
  534. mutex_unlock(&notifier_mutex);
  535. return 0;
  536. }
  537. bool audio_notifier_probe_status(void)
  538. {
  539. struct adsp_notify_private *priv = NULL;
  540. struct platform_device *pdev = NULL;
  541. if (!adsp_private)
  542. goto exit;
  543. pdev = adsp_private;
  544. priv = platform_get_drvdata(pdev);
  545. if (!priv) {
  546. dev_err(&pdev->dev," %s: Private data get failed\n", __func__);
  547. goto exit;
  548. }
  549. if (priv->notifier_probe_complete) {
  550. dev_dbg(&pdev->dev, "%s: audio notify probe successfully completed\n",
  551. __func__);
  552. return true;
  553. }
  554. exit:
  555. return false;
  556. }
  557. EXPORT_SYMBOL(audio_notifier_probe_status);
  558. static int audio_notify_probe(struct platform_device *pdev)
  559. {
  560. int ret = -EINVAL;
  561. struct adsp_notify_private *priv = NULL;
  562. struct property *prop;
  563. int size;
  564. phandle rproc_phandle;
  565. adsp_private = NULL;
  566. priv = devm_kzalloc(&pdev->dev, sizeof(*priv), GFP_KERNEL);
  567. if (!priv) {
  568. ret = -ENOMEM;
  569. return ret;
  570. }
  571. priv->notifier_probe_complete = false;
  572. platform_set_drvdata(pdev, priv);
  573. prop = of_find_property(pdev->dev.of_node, "qcom,rproc-handle", &size);
  574. if (!prop) {
  575. dev_err(&pdev->dev, "Missing remotproc handle\n");
  576. return ret;
  577. }
  578. rproc_phandle = be32_to_cpup(prop->value);
  579. priv->rproc_h = rproc_get_by_phandle(rproc_phandle);
  580. if (!priv->rproc_h) {
  581. dev_info_ratelimited(&pdev->dev, "remotproc handle NULL\n");
  582. ret = -EPROBE_DEFER;
  583. return ret;
  584. }
  585. adsp_private = pdev;
  586. audio_notifier_subsys_init();
  587. audio_notifier_init_service(AUDIO_NOTIFIER_PDR_SERVICE);
  588. /* Do not return error since PDR enablement is not critical */
  589. audio_notifier_late_init();
  590. priv->notifier_probe_complete = true;
  591. return 0;
  592. }
  593. static int audio_notify_remove(struct platform_device *pdev)
  594. {
  595. return 0;
  596. }
  597. static const struct of_device_id adsp_notify_dt_match[] = {
  598. { .compatible = "qcom,adsp-notify" },
  599. { }
  600. };
  601. MODULE_DEVICE_TABLE(of, adsp_notify_dt_match);
  602. static struct platform_driver adsp_notify_driver = {
  603. .driver = {
  604. .name = "adsp-notify",
  605. .owner = THIS_MODULE,
  606. .of_match_table = adsp_notify_dt_match,
  607. .suppress_bind_attrs = true,
  608. },
  609. .probe = audio_notify_probe,
  610. .remove = audio_notify_remove,
  611. };
  612. static int __init audio_notifier_init(void)
  613. {
  614. return platform_driver_register(&adsp_notify_driver);
  615. }
  616. module_init(audio_notifier_init);
  617. static void __exit audio_notifier_exit(void)
  618. {
  619. platform_driver_unregister(&adsp_notify_driver);
  620. }
  621. module_exit(audio_notifier_exit);
  622. MODULE_SOFTDEP("pre: qcom_q6v5_pas");
  623. MODULE_DESCRIPTION("Audio notifier driver");
  624. MODULE_LICENSE("GPL v2");